12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851385238533854385538563857385838593860386138623863386438653866386738683869387038713872387338743875387638773878387938803881388238833884388538863887388838893890389138923893389438953896389738983899390039013902390339043905390639073908390939103911391239133914391539163917391839193920392139223923392439253926392739283929393039313932393339343935393639373938393939403941394239433944394539463947394839493950395139523953395439553956395739583959396039613962396339643965396639673968396939703971397239733974397539763977397839793980398139823983398439853986398739883989399039913992399339943995399639973998399940004001400240034004400540064007400840094010401140124013401440154016401740184019402040214022402340244025402640274028402940304031403240334034403540364037403840394040404140424043404440454046404740484049405040514052405340544055405640574058405940604061406240634064406540664067406840694070407140724073407440754076407740784079408040814082408340844085408640874088408940904091409240934094409540964097409840994100410141024103410441054106410741084109411041114112411341144115411641174118411941204121412241234124412541264127412841294130413141324133413441354136413741384139414041414142414341444145414641474148414941504151415241534154415541564157415841594160416141624163416441654166416741684169417041714172417341744175417641774178417941804181418241834184418541864187418841894190419141924193419441954196419741984199420042014202420342044205420642074208420942104211421242134214421542164217421842194220422142224223422442254226422742284229423042314232423342344235423642374238423942404241424242434244424542464247424842494250425142524253425442554256425742584259426042614262426342644265426642674268426942704271427242734274427542764277427842794280428142824283428442854286428742884289429042914292429342944295429642974298429943004301430243034304430543064307430843094310431143124313431443154316431743184319432043214322432343244325432643274328432943304331433243334334433543364337433843394340434143424343434443454346434743484349435043514352435343544355435643574358435943604361436243634364436543664367436843694370437143724373437443754376437743784379438043814382438343844385438643874388438943904391439243934394439543964397439843994400440144024403440444054406440744084409441044114412441344144415441644174418441944204421442244234424442544264427442844294430443144324433443444354436443744384439444044414442444344444445444644474448444944504451445244534454445544564457445844594460446144624463446444654466446744684469 |
- ///|/ Copyright (c) Prusa Research 2016 - 2023 Lukáš Hejl @hejllukas, Pavel Mikuš @Godrak, Lukáš Matěna @lukasmatena, Vojtěch Bubník @bubnikv, Enrico Turri @enricoturri1966, Oleksandra Iushchenko @YuSanka, David Kocík @kocikdav, Roman Beránek @zavorka
- ///|/ Copyright (c) 2021 Justin Schuh @jschuh
- ///|/ Copyright (c) 2021 Ilya @xorza
- ///|/ Copyright (c) 2016 Joseph Lenox @lordofhyphens
- ///|/ Copyright (c) Slic3r 2014 - 2016 Alessandro Ranellucci @alranel
- ///|/ Copyright (c) 2015 Maksim Derbasov @ntfshard
- ///|/
- ///|/ PrusaSlicer is released under the terms of the AGPLv3 or higher
- ///|/
- #include "AABBTreeLines.hpp"
- #include "BridgeDetector.hpp"
- #include "ExPolygon.hpp"
- #include "Exception.hpp"
- #include "Flow.hpp"
- #include "GCode/ExtrusionProcessor.hpp"
- #include "KDTreeIndirect.hpp"
- #include "Line.hpp"
- #include "Point.hpp"
- #include "Polygon.hpp"
- #include "Polyline.hpp"
- #include "Print.hpp"
- #include "BoundingBox.hpp"
- #include "ClipperUtils.hpp"
- #include "ElephantFootCompensation.hpp"
- #include "Geometry.hpp"
- #include "I18N.hpp"
- #include "Layer.hpp"
- #include "MutablePolygon.hpp"
- #include "PrintBase.hpp"
- #include "PrintConfig.hpp"
- #include "Support/SupportMaterial.hpp"
- #include "Support/TreeSupport.hpp"
- #include "Surface.hpp"
- #include "Slicing.hpp"
- #include "SurfaceCollection.hpp"
- #include "Tesselate.hpp"
- #include "Thread.hpp"
- #include "TriangleMeshSlicer.hpp"
- #include "Utils.hpp"
- #include "Fill/FillAdaptive.hpp"
- #include "Fill/FillLightning.hpp"
- #include "Format/STL.hpp"
- #include "Support/SupportMaterial.hpp"
- #include "SupportSpotsGenerator.hpp"
- #include "TriangleSelectorWrapper.hpp"
- #include "format.hpp"
- #include "libslic3r.h"
- #include <algorithm>
- #include <atomic>
- #include <cmath>
- #include <cstddef>
- #include <cstdint>
- #include <float.h>
- #include <functional>
- #include <limits>
- #include <map>
- #include <oneapi/tbb/blocked_range.h>
- #include <oneapi/tbb/concurrent_vector.h>
- #include <oneapi/tbb/parallel_for.h>
- #include <string>
- #include <string_view>
- #include <tuple>
- #include <unordered_map>
- #include <unordered_set>
- #include <utility>
- #include <boost/log/trivial.hpp>
- #include <tbb/parallel_for.h>
- #include <vector>
- using namespace std::literals;
- // #define PRINT_OBJECT_TIMING
- #ifdef PRINT_OBJECT_TIMING
- // time limit for one ClipperLib operation (union / diff / offset), in ms
- #define PRINT_OBJECT_TIME_LIMIT_DEFAULT 50
- #include <boost/current_function.hpp>
- #include "Timer.hpp"
- #define PRINT_OBJECT_TIME_LIMIT_SECONDS(limit) Timing::TimeLimitAlarm time_limit_alarm(uint64_t(limit) * 1000000000l, BOOST_CURRENT_FUNCTION)
- #define PRINT_OBJECT_TIME_LIMIT_MILLIS(limit) Timing::TimeLimitAlarm time_limit_alarm(uint64_t(limit) * 1000000l, BOOST_CURRENT_FUNCTION)
- #else
- #define PRINT_OBJECT_TIME_LIMIT_SECONDS(limit) do {} while(false)
- #define PRINT_OBJECT_TIME_LIMIT_MILLIS(limit) do {} while(false)
- #endif // PRINT_OBJECT_TIMING
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- #define SLIC3R_DEBUG
- #endif
- // #define SLIC3R_DEBUG
- // Make assert active if SLIC3R_DEBUG
- #ifdef SLIC3R_DEBUG
- #undef NDEBUG
- #define DEBUG
- #define _DEBUG
- #include "SVG.hpp"
- #undef assert
- #include <cassert>
- #endif
- #include "SVG.hpp"
- namespace Slic3r {
- // Constructor is called from the main thread, therefore all Model / ModelObject / ModelIntance data are valid.
- PrintObject::PrintObject(Print* print, ModelObject* model_object, const Transform3d& trafo, PrintInstances&& instances) :
- PrintObjectBaseWithState(print, model_object),
- m_trafo(trafo)
- {
- // Compute centering offet to be applied to our meshes so that we work with smaller coordinates
- // requiring less bits to represent Clipper coordinates.
- // Snug bounding box of a rotated and scaled object by the 1st instantion, without the instance translation applied.
- // All the instances share the transformation matrix with the exception of translation in XY and rotation by Z,
- // therefore a bounding box from 1st instance of a ModelObject is good enough for calculating the object center,
- // snug height and an approximate bounding box in XY.
- BoundingBoxf3 bbox = model_object->raw_bounding_box();
- Vec3d bbox_center = bbox.center();
- // We may need to rotate the bbox / bbox_center from the original instance to the current instance.
- double z_diff = Geometry::rotation_diff_z(model_object->instances.front()->get_matrix(), instances.front().model_instance->get_matrix());
- if (std::abs(z_diff) > EPSILON) {
- auto z_rot = Eigen::AngleAxisd(z_diff, Vec3d::UnitZ());
- bbox = bbox.transformed(Transform3d(z_rot));
- bbox_center = (z_rot * bbox_center).eval();
- }
- // Center of the transformed mesh (without translation).
- m_center_offset = Point::new_scale(bbox_center.x(), bbox_center.y());
- // Size of the transformed mesh. This bounding may not be snug in XY plane, but it is snug in Z.
- m_size = (bbox.size() * (1. / SCALING_FACTOR)).cast<coord_t>();
- m_size.z() = coord_t(model_object->max_z() * (1. / SCALING_FACTOR));
- this->set_instances(std::move(instances));
- //create config hierarchy
- m_config.parent = &print->config();
- }
- PrintBase::ApplyStatus PrintObject::set_instances(PrintInstances&& instances)
- {
- for (PrintInstance& i : instances)
- // Add the center offset, which will be subtracted from the mesh when slicing.
- i.shift += m_center_offset;
- // Invalidate and set copies.
- PrintBase::ApplyStatus status = PrintBase::APPLY_STATUS_UNCHANGED;
- bool equal_length = instances.size() == m_instances.size();
- bool equal = equal_length && std::equal(instances.begin(), instances.end(), m_instances.begin(),
- [](const PrintInstance& lhs, const PrintInstance& rhs) { return lhs.model_instance == rhs.model_instance && lhs.shift == rhs.shift; });
- if (! equal) {
- status = PrintBase::APPLY_STATUS_CHANGED;
- if (m_print->invalidate_steps({ psSkirtBrim, psGCodeExport }) ||
- (! equal_length && m_print->invalidate_step(psWipeTower)))
- status = PrintBase::APPLY_STATUS_INVALIDATED;
- m_instances = std::move(instances);
- for (PrintInstance &i : m_instances)
- i.print_object = this;
- }
- return status;
- }
- std::vector<std::reference_wrapper<const PrintRegion>> PrintObject::all_regions() const
- {
- std::vector<std::reference_wrapper<const PrintRegion>> out;
- out.reserve(m_shared_regions->all_regions.size());
- for (const std::unique_ptr<Slic3r::PrintRegion> ®ion : m_shared_regions->all_regions)
- out.emplace_back(*region.get());
- return out;
- }
- // 1) Merges typed region slices into stInternal type.
- // 2) Increases an "extra perimeters" counter at region slices where needed.
- // 3) Generates perimeters, gap fills and fill regions (fill regions of type stInternal).
- void PrintObject::make_perimeters()
- {
- // prerequisites
- this->slice();
- if (! this->set_started(posPerimeters))
- return;
- m_print->set_status(objectstep_2_percent[PrintObjectStep::posPerimeters], _u8L("Generating perimeters"));
- m_print->secondary_status_counter_add_max(m_layers.size());
- BOOST_LOG_TRIVIAL(info) << "Generating perimeters..." << log_memory_info();
-
- // Revert the typed slices into untyped slices.
- if (m_typed_slices) {
- for (Layer *layer : m_layers) {
- layer->clear_fills();
- layer->restore_untyped_slices();
- m_print->throw_if_canceled();
- }
- m_typed_slices = false;
- }
- // compare each layer to the one below, and mark those slices needing
- // one additional inner perimeter, like the top of domed objects-
-
- // this algorithm makes sure that at least one perimeter is overlapping
- // but we don't generate any extra perimeter if fill density is zero, as they would be floating
- // inside the object - infill_only_where_needed should be the method of choice for printing
- // hollow objects
- for (size_t region_id = 0; region_id < this->num_printing_regions(); ++ region_id) {
- const PrintRegion ®ion = this->printing_region(region_id);
- if (!region.config().extra_perimeters || region.config().perimeters == 0 ||
- region.config().fill_density == 0 || this->layer_count() < 2) {
- continue;
- }
- // use an antomic idx instead of the range, to avoid a thread being very late because it's on the difficult layers.
- BOOST_LOG_TRIVIAL(debug) << "Generating extra perimeters for region " << region_id << " in parallel - start";
- Slic3r::parallel_for(size_t(0), m_layers.size() - 1,
- [this, ®ion, region_id](const size_t layer_idx) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- m_print->throw_if_canceled();
- LayerRegion &layerm = *m_layers[layer_idx]->get_region(region_id);
- const LayerRegion &upper_layerm = *m_layers[layer_idx+1]->get_region(region_id);
- const Polygons upper_layerm_polygons = to_polygons(upper_layerm.slices().surfaces);
- // Filter upper layer polygons in intersection_ppl by their bounding boxes?
- // my $upper_layerm_poly_bboxes= [ map $_->bounding_box, @{$upper_layerm_polygons} ];
- const double total_loop_length = total_length(upper_layerm_polygons);
- const coord_t perimeter_spacing = layerm.flow(frPerimeter).scaled_spacing();
- const Flow ext_perimeter_flow = layerm.flow(frExternalPerimeter);
- const coord_t ext_perimeter_width = ext_perimeter_flow.scaled_width();
- const coord_t ext_perimeter_spacing = ext_perimeter_flow.scaled_spacing();
- // slice_mutable is not const because slice.extra_perimeters is being incremented.
- for (Surface &slice_mutable : layerm.m_slices.surfaces) {
- const Surface &slice = slice_mutable;
- for (;;) {
- // compute the total thickness of perimeters
- const coord_t perimeters_thickness = ext_perimeter_width/2 + ext_perimeter_spacing/2
- + (region.config().perimeters-1 + slice.extra_perimeters) * perimeter_spacing;
- // define a critical area where we don't want the upper slice to fall into
- // (it should either lay over our perimeters or outside this area)
- const coord_t critical_area_depth = coord_t(perimeter_spacing * 1.5);
- const Polygons critical_area = diff(
- offset(slice.expolygon, float(- perimeters_thickness)),
- offset(slice.expolygon, float(- perimeters_thickness - critical_area_depth))
- );
- // check whether a portion of the upper slices falls inside the critical area
- const Polylines intersection = intersection_pl(to_polylines(upper_layerm_polygons), critical_area);
- // only add an additional loop if at least 30% of the slice loop would benefit from it
- if (total_length(intersection) <= total_loop_length*0.3)
- break;
- /*
- if (0) {
- require "Slic3r/SVG.pm";
- Slic3r::SVG::output(
- "extra.svg",
- no_arrows => 1,
- expolygons => union_ex($critical_area),
- polylines => [ map $_->split_at_first_point, map $_->p, @{$upper_layerm->slices} ],
- );
- }
- */
- ++ slice_mutable.extra_perimeters;
- }
- #ifdef DEBUG
- if (slice.extra_perimeters > 0)
- printf(" adding %d more perimeter(s) at layer %zu\n", slice.extra_perimeters, layer_idx);
- #endif
- }
- }
- );
- m_print->throw_if_canceled();
- BOOST_LOG_TRIVIAL(debug) << "Generating extra perimeters for region " << region_id << " in parallel - end";
- }
- BOOST_LOG_TRIVIAL(debug) << "Generating perimeters in parallel - start";
- Slic3r::parallel_for(size_t(0), m_layers.size(),
- [this](const size_t layer_idx) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- m_print->throw_if_canceled();
- // updating progress
- int32_t nb_layers_done = m_print->secondary_status_counter_increment();
- m_print->set_status( int((nb_layers_done * 100) / m_print->secondary_status_counter_get_max()), L("Generating perimeters: layer %s / %s"),
- { std::to_string(nb_layers_done), std::to_string(m_print->secondary_status_counter_get_max()) }, PrintBase::SlicingStatus::SECONDARY_STATE);
- // make perimeters
- m_layers[layer_idx]->make_perimeters();
- }
- );
- m_print->throw_if_canceled();
- BOOST_LOG_TRIVIAL(debug) << "Generating perimeters in parallel - end";
- if (print()->config().milling_diameter.size() > 0) {
- BOOST_LOG_TRIVIAL(debug) << "Generating milling post-process in parallel - start";
- Slic3r::parallel_for(size_t(0), m_layers.size(),
- [this](const size_t layer_idx) {
- m_print->throw_if_canceled();
- m_layers[layer_idx]->make_milling_post_process();
- }
- );
- m_print->throw_if_canceled();
- BOOST_LOG_TRIVIAL(debug) << "Generating milling post-process in parallel - end";
- }
- this->set_done(posPerimeters);
- }
- void PrintObject::prepare_infill()
- {
- if (!this->set_started(posPrepareInfill))
- return;
- m_print->set_status(objectstep_2_percent[PrintObjectStep::posPrepareInfill], L("Preparing infill"));
- if (m_print->objects().size() == 1) {
- m_print->set_status(0, "", PrintBase::SlicingStatus::DEFAULT | PrintBase::SlicingStatus::SECONDARY_STATE);
- } else {
- // detect (33%) -> 25 25
- // prepare layers (1%) -> 5 30
- // discover shells (40%) -> 30 60
- // process externals (12%) -> 15 75
- // Clean surfaces (1%) -> 5 80
- // Put bridges over sparse infill (12%) -> 15 95
- // Combine infill (1%) -> 5 100
- m_print->secondary_status_counter_add_max(100);
- }
- if (m_typed_slices) {
- // To improve robustness of detect_surfaces_type() when reslicing (working with typed slices), see GH issue #7442.
- // The preceding step (perimeter generator) only modifies extra_perimeters and the extra perimeters are only used by discover_vertical_shells()
- // with more than a single region. If this step does not use Surface::extra_perimeters or Surface::extra_perimeters is always zero, it is safe
- // to reset to the untyped slices before re-runnning detect_surfaces_type().
- for (Layer* layer : m_layers) {
- layer->restore_untyped_slices_no_extra_perimeters();
- m_print->throw_if_canceled();
- }
- }
- // This will assign a type (top/bottom/internal) to $layerm->slices.
- // Then the classifcation of $layerm->slices is transfered onto
- // the $layerm->fill_surfaces by clipping $layerm->fill_surfaces
- // by the cummulative area of the previous $layerm->fill_surfaces.
- if (m_print->objects().size() == 1) {
- m_print->set_status(0, L("Detect surfaces types"), {}, PrintBase::SlicingStatus::SECONDARY_STATE);
- } else {
- int32_t advancement_count = m_print->secondary_status_counter_increment(25);
- m_print->set_status(advancement_count * 100 / m_print->secondary_status_counter_get_max(), L("Process objects: %s / %s"),
- {std::to_string(advancement_count),
- std::to_string(m_print->secondary_status_counter_get_max())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- this->detect_surfaces_type();
- m_print->throw_if_canceled();
-
- // Decide what surfaces are to be filled.
- // Here the stTop / stBottomBridge / stBottom infill is turned to just stInternal if zero top / bottom infill layers are configured.
- // Also tiny stInternal surfaces are turned to stInternalSolid.
- BOOST_LOG_TRIVIAL(info) << "Preparing fill surfaces..." << log_memory_info();
- if (m_print->objects().size() > 1) {
- int32_t advancement_count = m_print->secondary_status_counter_increment(5);
- m_print->set_status(advancement_count * 100 / m_print->secondary_status_counter_get_max(), L("Process objects: %s / %s"),
- {std::to_string(advancement_count),
- std::to_string(m_print->secondary_status_counter_get_max())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- for (size_t layer_idx = 0; layer_idx < m_layers.size(); ++layer_idx) {
- Layer *layer = m_layers[layer_idx];
- if (m_print->objects().size() == 1) {
- m_print->set_status(int(25 + (5 * layer_idx / m_layers.size())),
- L("Prepare fill surfaces: layer %s / %s"),
- {std::to_string(layer_idx), std::to_string(m_layers.size())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- for (auto *region : layer->m_regions) {
- region->prepare_fill_surfaces();
- m_print->throw_if_canceled();
- }
- }
- // Add solid fills to ensure the shell vertical thickness.
- if (m_print->objects().size() == 1) {
- m_print->set_status(30, L("Discover shells"), {}, PrintBase::SlicingStatus::SECONDARY_STATE);
- } else {
- int32_t advancement_count = m_print->secondary_status_counter_increment(30);
- m_print->set_status(advancement_count * 100 / m_print->secondary_status_counter_get_max(), L("Process objects: %s / %s"),
- {std::to_string(advancement_count),
- std::to_string(m_print->secondary_status_counter_get_max())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- this->discover_vertical_shells();
- m_print->throw_if_canceled();
- // Debugging output.
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- for (size_t region_id = 0; region_id < this->num_printing_regions(); ++ region_id) {
- for (const Layer *layer : m_layers) {
- LayerRegion *layerm = layer->m_regions[region_id];
- layerm->export_region_slices_to_svg_debug("3_discover_vertical_shells-final");
- layerm->export_region_fill_surfaces_to_svg_debug("3_discover_vertical_shells-final");
- } // for each layer
- } // for each region
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- // this will detect bridges and reverse bridges
- // and rearrange top/bottom/internal surfaces
- // It produces enlarged overlapping bridging areas.
- //
- // 1) stBottomBridge / stBottom infill is grown by 3mm and clipped by the total infill area. Bridges are detected. The areas may overlap.
- // 2) stTop is grown by 3mm and clipped by the grown bottom areas. The areas may overlap.
- // 3) Clip the internal surfaces by the grown top/bottom surfaces.
- // 4) Merge surfaces with the same style. This will mostly get rid of the overlaps.
- //FIXME This does not likely merge surfaces, which are supported by a material with different colors, but same properties.
- if (m_print->objects().size() == 1) {
- m_print->set_status( 60, L("Process external surfaces"), {}, PrintBase::SlicingStatus::SECONDARY_STATE);
- } else {
- int32_t advancement_count = m_print->secondary_status_counter_increment(15);
- m_print->set_status(advancement_count * 100 / m_print->secondary_status_counter_get_max(), L("Process objects: %s / %s"),
- {std::to_string(advancement_count),
- std::to_string(m_print->secondary_status_counter_get_max())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- this->process_external_surfaces();
- m_print->throw_if_canceled();
- // Debugging output.
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- for (size_t region_id = 0; region_id < this->num_printing_regions(); ++ region_id) {
- for (const Layer *layer : m_layers) {
- LayerRegion *layerm = layer->m_regions[region_id];
- layerm->export_region_slices_to_svg_debug("3_process_external_surfaces-final");
- layerm->export_region_fill_surfaces_to_svg_debug("3_process_external_surfaces-final");
- } // for each layer
- } // for each region
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- // Detect, which fill surfaces are near external layers.
- // They will be split in internal and internal-solid surfaces.
- // The purpose is to add a configurable number of solid layers to support the TOP surfaces
- // and to add a configurable number of solid layers above the BOTTOM / BOTTOMBRIDGE surfaces
- // to close these surfaces reliably.
- //FIXME Vojtech: Is this a good place to add supporting infills below sloping perimeters?
- //note: only if not "ensure vertical shell" (which doesn't exist anymore)
- this->discover_horizontal_shells();
- m_print->throw_if_canceled();
- //as there is some too thin solid surface, please deleted them and merge all of the surfacesthat are contigous.
- if (m_print->objects().size() == 1) {
- m_print->set_status( 75, L("Clean surfaces"), {}, PrintBase::SlicingStatus::SECONDARY_STATE);
- } else {
- int32_t advancement_count = m_print->secondary_status_counter_increment(5);
- m_print->set_status(advancement_count * 100 / m_print->secondary_status_counter_get_max(), L("Process objects: %s / %s"),
- {std::to_string(advancement_count),
- std::to_string(m_print->secondary_status_counter_get_max())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- this->clean_surfaces();
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- for (size_t region_id = 0; region_id < this->num_printing_regions(); ++ region_id) {
- for (const Layer *layer : m_layers) {
- LayerRegion *layerm = layer->m_regions[region_id];
- layerm->export_region_slices_to_svg_debug("7_discover_horizontal_shells-final");
- layerm->export_region_fill_surfaces_to_svg_debug("7_discover_horizontal_shells-final");
- } // for each layer
- } // for each region
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- // Only active if config->infill_only_where_needed. This step trims the sparse infill,
- // so it acts as an internal support. It maintains all other infill types intact.
- // Here the internal surfaces and perimeters have to be supported by the sparse infill.
- //FIXME The surfaces are supported by a sparse infill, but the sparse infill is only as large as the area to support.
- // Likely the sparse infill will not be anchored correctly, so it will not work as intended.
- // Also one wishes the perimeters to be supported by a full infill.
- //m_print->set_status( 70, L("Clip surfaces"), {}, PrintBase::SlicingStatus::SECONDARY_STATE);
- //this->clip_fill_surfaces(); // infill_only_where_needed
- m_print->throw_if_canceled();
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- for (size_t region_id = 0; region_id < this->num_printing_regions(); ++ region_id) {
- for (const Layer *layer : m_layers) {
- LayerRegion *layerm = layer->m_regions[region_id];
- layerm->export_region_slices_to_svg_debug("8_clip_surfaces-final");
- layerm->export_region_fill_surfaces_to_svg_debug("8_clip_surfaces-final");
- } // for each layer
- } // for each region
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
-
- // the following step needs to be done before combination because it may need
- // to remove only half of the combined infill
- if (m_print->objects().size() == 1) {
- m_print->set_status( 80, L("Put bridges over sparse infill"), {}, PrintBase::SlicingStatus::SECONDARY_STATE);
- } else {
- int32_t advancement_count = m_print->secondary_status_counter_increment(15);
- m_print->set_status(advancement_count * 100 / m_print->secondary_status_counter_get_max(), L("Process objects: %s / %s"),
- {std::to_string(advancement_count),
- std::to_string(m_print->secondary_status_counter_get_max())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- this->bridge_over_infill();
- m_print->throw_if_canceled();
- this->replaceSurfaceType(stPosInternal | stDensSolid,
- stPosInternal | stDensSolid | stModOverBridge,
- stPosInternal | stDensSolid | stModBridge);
- m_print->throw_if_canceled();
- this->replaceSurfaceType(stPosTop | stDensSolid,
- stPosTop | stDensSolid | stModOverBridge,
- stPosInternal | stDensSolid | stModBridge);
- m_print->throw_if_canceled();
- this->replaceSurfaceType(stPosInternal | stDensSolid,
- stPosInternal | stDensSolid | stModOverBridge,
- stPosBottom | stDensSolid | stModBridge);
- m_print->throw_if_canceled();
- this->replaceSurfaceType(stPosTop | stDensSolid,
- stPosTop | stDensSolid | stModOverBridge,
- stPosBottom | stDensSolid | stModBridge);
- m_print->throw_if_canceled();
- // combine fill surfaces to honor the "infill every N layers" option
- if (m_print->objects().size() == 1) {
- m_print->set_status( 95, L("Combine infill"), {}, PrintBase::SlicingStatus::SECONDARY_STATE);
- } else {
- int32_t advancement_count = m_print->secondary_status_counter_increment(5);
- m_print->set_status(advancement_count * 100 / m_print->secondary_status_counter_get_max(), L("Process objects: %s / %s"),
- {std::to_string(advancement_count),
- std::to_string(m_print->secondary_status_counter_get_max())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- this->combine_infill();
- m_print->throw_if_canceled();
- // count the distance from the nearest top surface, to allow to use denser infill
- // if needed and if infill_dense_layers is positive.
- this->tag_under_bridge();
- m_print->throw_if_canceled();
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- for (size_t region_id = 0; region_id < this->num_printing_regions(); ++ region_id) {
- for (const Layer *layer : m_layers) {
- LayerRegion *layerm = layer->m_regions[region_id];
- layerm->export_region_slices_to_svg_debug("9_prepare_infill-final");
- layerm->export_region_fill_surfaces_to_svg_debug("9_prepare_infill-final");
- } // for each layer
- } // for each region
- for (const Layer *layer : m_layers) {
- layer->export_region_slices_to_svg_debug("9_prepare_infill-final");
- layer->export_region_fill_surfaces_to_svg_debug("9_prepare_infill-final");
- } // for each layer
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- //compute m_max_sparse_spacing for fill_aligned_z
- _compute_max_sparse_spacing();
-
- if (m_print->objects().size() > 1) {
- int32_t advancement_count = m_print->secondary_status_counter_increment(0);
- m_print->set_status(advancement_count * 100 / m_print->secondary_status_counter_get_max(), L("Process objects: %s / %s"),
- {std::to_string(advancement_count),
- std::to_string(m_print->secondary_status_counter_get_max())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- this->set_done(posPrepareInfill);
- }
- void PrintObject::_compute_max_sparse_spacing()
- {
- m_max_sparse_spacing = 0;
- std::atomic_int64_t max_sparse_spacing(0);
- Slic3r::parallel_for(size_t(0), m_layers.size(),
- [this, &max_sparse_spacing](const size_t layer_idx) {
- m_print->throw_if_canceled();
- const Layer *layer = m_layers[layer_idx];
- for (const LayerRegion *layerm : layer->regions()) {
- // check if region has sparse infill.
- for (const Surface &surface : layerm->fill_surfaces().surfaces) {
- if (surface.has_fill_sparse()) {
- coord_t spacing = layerm->region().flow(*this, frInfill, layer->height, layer->id()).scaled_spacing();
- // update atomic to max
- int64_t prev_value = max_sparse_spacing.load();
- while (prev_value < int64_t(spacing) &&
- !max_sparse_spacing.compare_exchange_weak(prev_value, int64_t(spacing))) {
- }
- }
- }
- }
- });
- m_max_sparse_spacing = max_sparse_spacing.load();
- }
- void PrintObject::clear_fills()
- {
- for (Layer *layer : m_layers)
- layer->clear_fills();
- }
- void PrintObject::infill()
- {
- // prerequisites
- this->prepare_infill();
- //m_print->set_status(0, _u8L("Infilling layer %s / %s"),
- // { std::to_string(0), std::to_string(m_layers.size()) }, PrintBase::SlicingStatus::SECONDARY_STATE);
- if (this->set_started(posInfill)) {
- // TRN Status for the Print calculation
- m_print->set_status(objectstep_2_percent[PrintObjectStep::posInfill], L("Infilling layers"));
- m_print->secondary_status_counter_add_max(m_layers.size());
- const auto& adaptive_fill_octree = this->m_adaptive_fill_octrees.first;
- const auto& support_fill_octree = this->m_adaptive_fill_octrees.second;
- BOOST_LOG_TRIVIAL(debug) << "Filling layers in parallel - start";
- Slic3r::parallel_for(size_t(0), m_layers.size(),
- [this, &adaptive_fill_octree = adaptive_fill_octree, &support_fill_octree = support_fill_octree]
- (const size_t layer_idx) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- // updating progress
- int32_t nb_layers_done = m_print->secondary_status_counter_increment();
- m_print->set_status(100 * nb_layers_done / m_print->secondary_status_counter_get_max(), L("Infilling layer %s / %s"),
- {std::to_string(nb_layers_done), std::to_string(m_print->secondary_status_counter_get_max())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- std::chrono::time_point<std::chrono::system_clock> start_make_fill = std::chrono::system_clock::now();
- m_print->throw_if_canceled();
- m_layers[layer_idx]->make_fills(adaptive_fill_octree.get(), support_fill_octree.get(), this->m_lightning_generator.get());
- }
- );
- m_print->set_status(100, "", PrintBase::SlicingStatus::SECONDARY_STATE);
- m_print->throw_if_canceled();
- BOOST_LOG_TRIVIAL(debug) << "Filling layers in parallel - end";
- /* we could free memory now, but this would make this step not idempotent
- ### $_->fill_surfaces->clear for map @{$_->regions}, @{$object->layers};
- */
- this->set_done(posInfill);
- }
- }
- void PrintObject::ironing()
- {
- if (this->set_started(posIroning)) {
- m_print->set_status(objectstep_2_percent[PrintObjectStep::posIroning], L("Ironing"));
- m_print->secondary_status_counter_add_max(m_layers.size());
- BOOST_LOG_TRIVIAL(debug) << "Ironing in parallel - start";
- // Ironing starting with layer 0 to support ironing all surfaces.
- Slic3r::parallel_for(size_t(0), m_layers.size(),
- [this](const size_t layer_idx) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- // updating progress
- int32_t nb_layers_done = m_print->secondary_status_counter_increment();
- m_print->set_status(100 * nb_layers_done / m_print->secondary_status_counter_get_max(), L("Ironing layer %s / %s"),
- {std::to_string(nb_layers_done), std::to_string(m_print->secondary_status_counter_get_max())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- m_print->throw_if_canceled();
- m_layers[layer_idx]->make_ironing();
- }
- );
- m_print->throw_if_canceled();
- BOOST_LOG_TRIVIAL(debug) << "Ironing in parallel - end";
- this->set_done(posIroning);
- }
- }
- void PrintObject::generate_support_spots()
- {
- assert(this->default_region_config(this->print()->default_region_config()).get_computed_value("perimeter_acceleration") > -1);
- if (this->set_started(posSupportSpotsSearch)) {
- BOOST_LOG_TRIVIAL(debug) << "Searching support spots - start";
- m_print->set_status(objectstep_2_percent[PrintObjectStep::posSupportSpotsSearch], L("Searching support spots"));
- if (m_print->objects().size() > 1) {
- m_print->secondary_status_counter_add_max(1);
- m_print->set_status(0. / m_print->objects().size(), L("Object %s / %s"),
- {std::to_string(0), std::to_string(m_print->objects().size())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- } else {
- m_print->set_status(0, "", PrintBase::SlicingStatus::DEFAULT | PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- if (!this->shared_regions()->generated_support_points.has_value()) {
- PrintTryCancel cancel_func = m_print->make_try_cancel();
- const PrintRegionConfig ®ion_config = this->default_region_config(this->print()->default_region_config());
- SupportSpotsGenerator::Params params{this->print()->m_config.filament_type.get_values(),
- float(region_config.get_computed_value("perimeter_acceleration")),
- this->config().raft_layers.value,
- float(this->config().brim_width.value),
- float(this->config().brim_width_interior.value)};
- auto [supp_points, partial_objects] = SupportSpotsGenerator::full_search(this, cancel_func, params);
- Transform3d po_transform = this->trafo_centered();
- if (this->layer_count() > 0) {
- po_transform = Geometry::translation_transform(Vec3d{0, 0, this->layers().front()->bottom_z()}) * po_transform;
- }
- this->m_shared_regions->generated_support_points = {po_transform, supp_points, partial_objects};
- m_print->throw_if_canceled();
- }
- // updating progress
- if (m_print->objects().size() > 1) {
- int32_t nb_objects_done = m_print->secondary_status_counter_increment();
- m_print->set_status(100 * (nb_objects_done + 1) / m_print->secondary_status_counter_get_max(),
- L("Object %s / %s"),
- {std::to_string(nb_objects_done + 1), std::to_string(m_print->secondary_status_counter_get_max())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- BOOST_LOG_TRIVIAL(debug) << "Searching support spots - end";
- this->set_done(posSupportSpotsSearch);
- }
- }
- void PrintObject::generate_support_material()
- {
- if (this->set_started(posSupportMaterial)) {
- m_print->set_status(objectstep_2_percent[PrintObjectStep::posSupportMaterial], L("Generating support material"));
- if (m_print->objects().size() > 1) {
- m_print->secondary_status_counter_add_max(1);
- m_print->set_status(0. / m_print->objects().size(), L("Object %s / %s"),
- {std::to_string(0), std::to_string(m_print->objects().size())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- } else {
- m_print->set_status(0, "", PrintBase::SlicingStatus::DEFAULT | PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- this->clear_support_layers();
- if ((this->has_support() && m_layers.size() > 1) || (this->has_raft() && ! m_layers.empty())) {
- this->_generate_support_material();
- m_print->throw_if_canceled();
- } else {
- #if 0
- // Printing without supports. Empty layer means some objects or object parts are levitating,
- // therefore they cannot be printed without supports.
- for (const Layer *layer : m_layers)
- if (layer->empty())
- throw Slic3r::SlicingError("Levitating objects cannot be printed without supports.");
- #endif
- }
- this->set_done(posSupportMaterial);
-
- // updating progress
- if (m_print->objects().size() > 1) {
- int32_t nb_objects_done = m_print->secondary_status_counter_increment();
- m_print->set_status(100 * (nb_objects_done + 1) / m_print->secondary_status_counter_get_max(),
- L("Object %s / %s"),
- {std::to_string(nb_objects_done + 1), std::to_string(m_print->secondary_status_counter_get_max())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- }
- }
- void PrintObject::simplify_extrusion_path()
- {
- if (this->set_started(posSimplifyPath)) {
- const PrintConfig& print_config = this->print()->config();
- const bool spiral_mode = print_config.spiral_vase;
- const bool enable_arc_fitting = print_config.arc_fitting != ArcFittingType::Disabled && !spiral_mode;
- m_print->secondary_status_counter_add_max(m_layers.size() + m_support_layers.size());
- BOOST_LOG_TRIVIAL(debug) << "Simplify extrusion path of object in parallel - start";
- //BBS: infill and walls
- Slic3r::parallel_for(size_t(0), m_layers.size(),
- [this](const size_t layer_idx) {
- m_print->throw_if_canceled();
- m_layers[layer_idx]->simplify_extrusion_path();
-
- // updating progress
- int32_t nb_layers_done = m_print->secondary_status_counter_increment() + 1;
- m_print->set_status(int((nb_layers_done * 100) / m_print->secondary_status_counter_get_max()),
- L("Optimizing layer %s / %s"),
- {std::to_string(nb_layers_done), std::to_string(m_print->secondary_status_counter_get_max())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- );
- //also simplify object skirt & brim
- if (enable_arc_fitting) {
- coordf_t scaled_resolution = scale_d(print_config.arc_fitting_resolution.get_abs_value(print_config.resolution.value));
- if (scaled_resolution == 0) scaled_resolution = enable_arc_fitting ? SCALED_EPSILON * 2 : SCALED_EPSILON;
- const ConfigOptionFloatOrPercent& arc_fitting_tolerance = print_config.arc_fitting_tolerance;
- GetPathsVisitor visitor;
- this->m_skirt.visit(visitor);
- this->m_brim.visit(visitor);
- tbb::parallel_for(
- tbb::blocked_range<size_t>(0, visitor.paths.size() + visitor.paths3D.size()),
- [this, &visitor, scaled_resolution, &arc_fitting_tolerance, &print_config](const tbb::blocked_range<size_t>& range) {
- size_t path_idx = range.begin();
- for (; path_idx < range.end() && path_idx < visitor.paths.size(); ++path_idx) {
- visitor.paths[path_idx]->simplify(scaled_resolution, print_config.arc_fitting, arc_fitting_tolerance.get_abs_value(visitor.paths[path_idx]->width()));
- }
- for (; path_idx < range.end() && path_idx - visitor.paths.size() < visitor.paths3D.size(); ++path_idx) {
- visitor.paths3D[path_idx - visitor.paths.size()]->simplify(scaled_resolution, print_config.arc_fitting,
- arc_fitting_tolerance.get_abs_value(visitor.paths3D[path_idx - visitor.paths.size()]->width()));
- }
- }
- );
- }
- m_print->throw_if_canceled();
- BOOST_LOG_TRIVIAL(debug) << "Simplify extrusion path of object in parallel - end";
- //BBS: share same progress
- BOOST_LOG_TRIVIAL(debug) << "Simplify extrusion path of support in parallel - start";
- Slic3r::parallel_for(size_t(0), m_support_layers.size(),
- [this](const size_t layer_idx) {
- m_print->throw_if_canceled();
- m_support_layers[layer_idx]->simplify_support_extrusion_path();
- // updating progress
- int32_t nb_layers_done = m_print->secondary_status_counter_increment() + 1;
- m_print->set_status(int((nb_layers_done * 100) / m_print->secondary_status_counter_get_max()),
- L("Optimizing layer %s / %s"),
- {std::to_string(nb_layers_done), std::to_string(m_print->secondary_status_counter_get_max())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- );
- m_print->throw_if_canceled();
- BOOST_LOG_TRIVIAL(debug) << "Simplify extrusion path of support in parallel - end";
- this->set_done(posSimplifyPath);
- }
- }
- void PrintObject::estimate_curled_extrusions()
- {
- if (this->set_started(posEstimateCurledExtrusions)) {
- m_print->set_status(objectstep_2_percent[PrintObjectStep::posEstimateCurledExtrusions], L("Estimate curled extrusions"));
- if (m_print->objects().size() > 1) {
- m_print->secondary_status_counter_add_max(1);
- m_print->set_status(0. / m_print->objects().size(), L("Object %s / %s"),
- {std::to_string(0), std::to_string(m_print->objects().size())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- } else {
- m_print->set_status(0, "", PrintBase::SlicingStatus::DEFAULT | PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- if (this->print()->config().avoid_crossing_curled_overhangs ||
- std::any_of(this->print()->m_print_regions.begin(), this->print()->m_print_regions.end(),
- [](const PrintRegion *region) { return region->config().overhangs_dynamic_speed.is_enabled(); })) {
- BOOST_LOG_TRIVIAL(debug) << "Estimating areas with curled extrusions - start";
- m_print->set_status(objectstep_2_percent[PrintObjectStep::posEstimateCurledExtrusions], _u8L("Estimating curled extrusions"));
- // Estimate curling of support material and add it to the malformaition lines of each layer
- float support_flow_width = support_material_flow(this, this->config().layer_height).width();
- SupportSpotsGenerator::Params params{this->print()->m_config.filament_type.get_values(),
- float(this->print()->full_print_config().get_computed_value("perimeter_acceleration")),
- this->config().raft_layers.value,
- float(this->config().brim_width.value),
- float(this->config().brim_width_interior.value)};
- SupportSpotsGenerator::estimate_supports_malformations(this->edit_support_layers(), support_flow_width, params);
- SupportSpotsGenerator::estimate_malformations(this->layers(), params);
- m_print->throw_if_canceled();
- BOOST_LOG_TRIVIAL(debug) << "Estimating areas with curled extrusions - end";
- }
- this->set_done(posEstimateCurledExtrusions);
-
- // updating progress
- if (m_print->objects().size() > 1) {
- int32_t nb_objects_done = m_print->secondary_status_counter_increment();
- m_print->set_status(100 * (nb_objects_done + 1) / m_print->secondary_status_counter_get_max(),
- L("Object %s / %s"),
- {std::to_string(nb_objects_done + 1), std::to_string(m_print->secondary_status_counter_get_max())},
- PrintBase::SlicingStatus::SECONDARY_STATE);
- }
- }
- }
- void PrintObject::calculate_overhanging_perimeters()
- {
- if (this->set_started(posCalculateOverhangingPerimeters)) {
- BOOST_LOG_TRIVIAL(debug) << "Calculating overhanging perimeters - start";
- m_print->set_status(objectstep_2_percent[PrintObjectStep::posCalculateOverhangingPerimeters], _u8L("Calculating overhanging perimeters"));
- std::set<uint16_t> extruders;
- std::unordered_set<const PrintRegion *> regions_with_dynamic_speeds;
- for (const PrintRegion *pr : this->print()->m_print_regions) {
- if (pr->config().overhangs_dynamic_speed.is_enabled()) {
- regions_with_dynamic_speeds.insert(pr);
- }
- extruders.clear();
- pr->collect_object_printing_extruders(*this->print(), extruders);
- auto cfg = this->print()->config();
- if (std::any_of(extruders.begin(), extruders.end(),
- [&cfg](unsigned int extruder_id) { return cfg.overhangs_dynamic_fan_speed.is_enabled(extruder_id); })) {
- regions_with_dynamic_speeds.insert(pr);
- }
- }
- if (!regions_with_dynamic_speeds.empty()) {
- std::unordered_map<size_t, AABBTreeLines::LinesDistancer<CurledLine>> curled_lines;
- std::unordered_map<size_t, AABBTreeLines::LinesDistancer<Linef>> unscaled_polygons_lines;
- for (const Layer *l : this->layers()) {
- curled_lines[l->id()] = AABBTreeLines::LinesDistancer<CurledLine>{l->curled_lines};
- unscaled_polygons_lines[l->id()] = AABBTreeLines::LinesDistancer<Linef>{to_unscaled_linesf(l->lslices())};
- }
- curled_lines[size_t(-1)] = {};
- unscaled_polygons_lines[size_t(-1)] = {};
- Slic3r::parallel_for(size_t(0), m_layers.size(),
- [this, &curled_lines, &unscaled_polygons_lines, ®ions_with_dynamic_speeds]
- (const size_t layer_idx) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- auto l = m_layers[layer_idx];
- // first layer: do not split
- if (l->id() > 0) {
- for (LayerRegion *layer_region : l->regions()) {
- if (regions_with_dynamic_speeds.find(layer_region->m_region) == regions_with_dynamic_speeds.end()) {
- continue;
- }
- size_t prev_layer_id = l->lower_layer ? l->lower_layer->id() : size_t(-1);
- const double nozzle_diameter_overhangs = layer_region->bridging_flow(frPerimeter).nozzle_diameter();
- double max_width = -1;
- if (layer_region->region().config().overhangs_width_speed.is_enabled()) {
- max_width = layer_region->region().config().overhangs_width_speed.get_abs_value(nozzle_diameter_overhangs);
- }
- if (layer_region->region().config().overhangs_width.is_enabled() && (max_width < 0 ||
- max_width > layer_region->region().config().overhangs_width.get_abs_value(nozzle_diameter_overhangs))) {
- max_width = layer_region->region().config().overhangs_width.get_abs_value(nozzle_diameter_overhangs);
- }
- if (max_width < 0) {
- max_width = nozzle_diameter_overhangs;
- }
- layer_region->m_perimeters =
- ExtrusionProcessor::calculate_and_split_overhanging_extrusions(&layer_region->m_perimeters,
- unscaled_polygons_lines[prev_layer_id],
- curled_lines[l->id()],
- max_width);
- }
- #ifdef _DEBUG
- {
- struct OverhangAssertVisitor : public ExtrusionVisitorRecursiveConst
- {
- virtual void default_use(const ExtrusionEntity &entity) override{};
- virtual void use(const ExtrusionPath &path) override {
- if (path.role().is_overhang())
- assert(path.attributes().overhang_attributes.has_value());
- }
- };
- OverhangAssertVisitor ov_visitor;
- for (auto ® : l->regions()) {
- reg->perimeters().visit(ov_visitor);
- }
- }
- #endif
- }
- });
- m_print->throw_if_canceled();
- BOOST_LOG_TRIVIAL(debug) << "Calculating overhanging perimeters - end";
- }
- this->set_done(posCalculateOverhangingPerimeters);
- }
- }
- std::pair<FillAdaptive::OctreePtr, FillAdaptive::OctreePtr> PrintObject::prepare_adaptive_infill_data(
- const std::vector<std::pair<const Surface *, float>> &surfaces_w_bottom_z) const
- {
- using namespace FillAdaptive;
- auto [adaptive_line_spacing, support_line_spacing] = adaptive_fill_line_spacing(*this);
- if ((adaptive_line_spacing == 0. && support_line_spacing == 0.) || this->layers().empty())
- return std::make_pair(OctreePtr(), OctreePtr());
- indexed_triangle_set mesh = this->model_object()->raw_indexed_triangle_set();
- // Rotate mesh and build octree on it with axis-aligned (standart base) cubes.
- auto to_octree = transform_to_octree().toRotationMatrix();
- its_transform(mesh, to_octree * this->trafo_centered(), true);
- // Triangulate internal bridging surfaces.
- std::vector<std::vector<Vec3d>> overhangs(std::max(surfaces_w_bottom_z.size(), size_t(1)));
- // ^ make sure vector is not empty, even with no briding surfaces we still want to build the adaptive trees later, some continue normally
- Slic3r::parallel_for(size_t(0), surfaces_w_bottom_z.size(),
- [this, &to_octree, &overhangs, &surfaces_w_bottom_z] (const size_t surface_idx) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- std::vector<Vec3d> &out = overhangs[surface_idx];
- m_print->throw_if_canceled();
- append(out, triangulate_expolygon_3d(surfaces_w_bottom_z[surface_idx].first->expolygon,
- surfaces_w_bottom_z[surface_idx].second));
- for (Vec3d &p : out)
- p = (to_octree * p).eval();
- }
- );
- // and gather them.
- for (size_t i = 1; i < overhangs.size(); ++ i)
- append(overhangs.front(), std::move(overhangs[i]));
- return std::make_pair(
- adaptive_line_spacing ? build_octree(mesh, overhangs.front(), adaptive_line_spacing, false) : OctreePtr(),
- support_line_spacing ? build_octree(mesh, overhangs.front(), support_line_spacing, true) : OctreePtr());
- }
- FillLightning::GeneratorPtr PrintObject::prepare_lightning_infill_data()
- {
- bool has_lightning_infill = false;
- coordf_t lightning_density = 0.;
- size_t lightning_cnt = 0;
- for (size_t region_id = 0; region_id < this->num_printing_regions(); ++region_id)
- if (const PrintRegionConfig &config = this->printing_region(region_id).config(); config.fill_density > 0 && config.fill_pattern.value == ipLightning) {
- has_lightning_infill = true;
- lightning_density += config.fill_density;
- ++lightning_cnt;
- }
- if (has_lightning_infill)
- lightning_density /= coordf_t(lightning_cnt);
- return has_lightning_infill ? FillLightning::build_generator(std::as_const(*this), lightning_density, [this]() -> void { this->throw_if_canceled(); }) : FillLightning::GeneratorPtr();
- }
- const PrintRegionConfig &PrintObject::default_region_config(const PrintRegionConfig &from_print) const {
- //TODO check if a regionconfig set in an object modifier go through
- if (this->m_shared_regions && num_printing_regions() > 0) {
- return printing_region(0).config();
- }
- return from_print;
- }
- bool PrintObject::has_brim() const {
- bool has_brim_volume = false;
- for (const ModelVolume *volume : this->model_object()->volumes) {
- if (volume->is_brim_patch()) {
- has_brim_volume = true;
- }
- }
- return has_brim_volume || ((this->config().brim_width.value > 0 && this->config().brim_width_interior.value > 0)
- && !this->has_raft());
- }
- Polygons PrintObject::get_brim_patch(ModelVolumeType brim_type, const PrintInstance *instance /*= nullptr*/) const {
- Polygons polys;
- for (const ModelVolume *v : this->model_object()->volumes) {
- assert(v);
- if (v->type() == brim_type) {
- if (instance == nullptr) {
- for (const PrintInstance &inst : this->instances()) {
- Polygons vol_outline;
- auto transl = Transform3d::Identity();
- assert(inst.model_instance);
- vol_outline = project_mesh(v->mesh().its,
- transl * inst.model_instance->get_matrix() * v->get_matrix(), [] {});
- append(polys, vol_outline);
- }
- } else {
- Polygons vol_outline;
- auto transl = Transform3d::Identity();
- assert(instance->model_instance);
- vol_outline = project_mesh(v->mesh().its,
- transl * instance->model_instance->get_matrix() * v->get_matrix(), [] {});
- append(polys, vol_outline);
- }
- }
- }
- coord_t scaled_brim_resolution = std::max(SCALED_EPSILON * 10, scale_t(this->print()->config().resolution.value));
- return ensure_valid(union_(polys), scaled_brim_resolution);
- }
- void PrintObject::clear_layers()
- {
- for (Layer *l : m_layers)
- delete l;
- m_layers.clear();
- }
- Layer* PrintObject::add_layer(int id, coordf_t height, coordf_t print_z, coordf_t slice_z)
- {
- m_layers.emplace_back(new Layer(id, this, height, print_z, slice_z));
- return m_layers.back();
- }
- void PrintObject::clear_support_layers()
- {
- for (Layer *l : m_support_layers)
- delete l;
- m_support_layers.clear();
- }
- void PrintObject::add_support_layer(int id, int interface_id, coordf_t height, coordf_t print_z)
- {
- m_support_layers.emplace_back(new SupportLayer(id, interface_id, this, height, print_z, -1));
- }
- SupportLayerPtrs::iterator PrintObject::insert_support_layer(SupportLayerPtrs::const_iterator pos, size_t id, size_t interface_id, coordf_t height, coordf_t print_z, coordf_t slice_z)
- {
- return m_support_layers.insert(pos, new SupportLayer(id, interface_id, this, height, print_z, slice_z));
- }
- // Called by Print::apply().
- // This method only accepts PrintObjectConfig and PrintRegionConfig option keys.
- bool PrintObject::invalidate_state_by_config_options(
- const ConfigOptionResolver &old_config, const ConfigOptionResolver &new_config, const std::vector<t_config_option_key> &opt_keys)
- {
- if (opt_keys.empty())
- return false;
- std::vector<PrintObjectStep> steps;
- bool invalidated = false;
- for (const t_config_option_key& opt_key : opt_keys) {
- if (
- opt_key == "arc_fitting"
- || opt_key == "external_perimeters_first"
- || opt_key == "external_perimeters_first_force"
- || opt_key == "external_perimeters_hole"
- || opt_key == "external_perimeters_nothole"
- || opt_key == "external_perimeter_extrusion_change_odd_layers"
- || opt_key == "external_perimeter_extrusion_spacing"
- || opt_key == "external_perimeter_extrusion_width"
- || opt_key == "external_perimeters_vase"
- || opt_key == "gap_fill_extension"
- || opt_key == "gap_fill_last"
- || opt_key == "gap_fill_max_width"
- || opt_key == "gap_fill_min_area"
- || opt_key == "gap_fill_min_length"
- || opt_key == "gap_fill_min_width"
- || opt_key == "min_width_top_surface"
- || opt_key == "only_one_perimeter_first_layer"
- || opt_key == "only_one_perimeter_top"
- || opt_key == "only_one_perimeter_top_other_algo"
- || opt_key == "overhangs_dynamic_speed"
- || opt_key == "overhangs_reverse"
- || opt_key == "overhangs_reverse_threshold"
- || opt_key == "overhangs_speed_enforce"
- || opt_key == "overhangs_width_speed"
- || opt_key == "overhangs_width"
- || opt_key == "perimeter_bonding"
- || opt_key == "perimeter_direction"
- || opt_key == "perimeter_extrusion_change_odd_layers"
- || opt_key == "perimeter_extrusion_spacing"
- || opt_key == "perimeter_extrusion_width"
- || opt_key == "perimeter_loop"
- || opt_key == "perimeter_loop_seam"
- || opt_key == "perimeter_reverse"
- || opt_key == "perimeter_round_corners"
- || opt_key == "thin_perimeters"
- || opt_key == "thin_perimeters_all"
- || opt_key == "thin_walls_merge"
- || opt_key == "thin_walls_min_width"
- || opt_key == "thin_walls_overlap"
- ) {
- steps.emplace_back(posPerimeters);
- } else if (
- opt_key == "gap_fill_enabled"
- || opt_key == "gap_fill_speed") {
- // Return true if gap-fill speed has changed from zero value to non-zero or from non-zero value to zero.
- auto is_gap_fill_changed_state_due_to_speed = [&opt_key, &old_config, &new_config]() -> bool {
- if (opt_key == "gap_fill_speed") {
- assert(old_config.option<ConfigOptionFloatOrPercent>(opt_key) && new_config.option<ConfigOptionFloatOrPercent>(opt_key));
- const float old_gap_fill_speed = old_config.option(opt_key)->get_float();
- const float new_gap_fill_speed = new_config.option(opt_key)->get_float();
- return (old_gap_fill_speed > 0.f && new_gap_fill_speed == 0.f) ||
- (old_gap_fill_speed == 0.f && new_gap_fill_speed > 0.f);
- }
- return false;
- };
- // Filtering of unprintable regions in multi-material segmentation depends on if gap-fill is enabled or not.
- // So step posSlice is invalidated when gap-fill was enabled/disabled by option "gap_fill_enabled" or by
- // changing "gap_fill_speed" to force recomputation of the multi-material segmentation.
- if (this->is_mm_painted() && (opt_key == "gap_fill_enabled" || (opt_key == "gap_fill_speed" && is_gap_fill_changed_state_due_to_speed())))
- steps.emplace_back(posSlice);
- steps.emplace_back(posPerimeters);
- } else if (
- // || opt_key == "exact_last_layer_height"
- opt_key == "bridge_type"
- || opt_key == "clip_multipart_objects"
- || opt_key == "curve_smoothing_angle_concave"
- || opt_key == "curve_smoothing_angle_convex"
- || opt_key == "curve_smoothing_cutoff_dist"
- || opt_key == "curve_smoothing_precision"
- || opt_key == "dont_support_bridges"
- || opt_key == "elephant_foot_min_width" //sla ?
- || opt_key == "first_layer_size_compensation"
- || opt_key == "first_layer_size_compensation_layers"
- || opt_key == "first_layer_size_compensation_no_collapse"
- || opt_key == "first_layer_height"
- || opt_key == "hole_size_compensation"
- || opt_key == "hole_size_threshold"
- || opt_key == "hole_to_polyhole"
- || opt_key == "hole_to_polyhole_threshold"
- || opt_key == "hole_to_polyhole_twisted"
- || opt_key == "layer_height"
- || opt_key == "min_bead_width"
- || opt_key == "min_feature_size"
- || opt_key == "mmu_segmented_region_max_width"
- || opt_key == "model_precision"
- || opt_key == "overhangs_max_slope"
- || opt_key == "overhangs_bridge_threshold"
- || opt_key == "overhangs_bridge_upper_layers"
- || opt_key == "raft_contact_distance"
- || opt_key == "raft_interface_layer_height"
- || opt_key == "raft_layers"
- || opt_key == "raft_layer_height"
- || opt_key == "perimeter_generator"
- || opt_key == "slice_closing_radius"
- || opt_key == "slicing_mode"
- || opt_key == "support_material_contact_distance_type"
- || opt_key == "support_material_contact_distance"
- || opt_key == "support_material_bottom_contact_distance"
- || opt_key == "support_material_interface_layer_height"
- || opt_key == "support_material_layer_height"
- || opt_key == "wall_transition_length"
- || opt_key == "wall_transition_filter_deviation"
- || opt_key == "wall_transition_angle"
- || opt_key == "wall_distribution_count"
- || opt_key == "xy_inner_size_compensation"
- || opt_key == "xy_size_compensation") {
- steps.emplace_back(posSlice);
- } else if (opt_key == "support_material") {
- steps.emplace_back(posSupportMaterial);
- if (m_config.support_material_contact_distance.value == 0. || m_config.support_material_bottom_contact_distance.value == 0.) {
- // Enabling / disabling supports while soluble support interface is enabled.
- // This changes the bridging logic (bridging enabled without supports, disabled with supports).
- // Reset everything.
- // See GH #1482 for details.
- steps.emplace_back(posSlice);
- }
- } else if (
- opt_key == "raft_expansion"
- || opt_key == "raft_first_layer_density"
- || opt_key == "raft_first_layer_expansion"
- || opt_key == "support_material_auto"
- || opt_key == "support_material_angle"
- || opt_key == "support_material_angle_height"
- || opt_key == "support_material_buildplate_only"
- || opt_key == "support_material_enforce_layers"
- || opt_key == "support_material_extruder"
- || opt_key == "support_material_extrusion_width"
- || opt_key == "support_material_interface_layers"
- || opt_key == "support_material_bottom_interface_layers"
- || opt_key == "support_material_bottom_interface_pattern"
- || opt_key == "support_material_interface_angle"
- || opt_key == "support_material_interface_angle_increment"
- || opt_key == "support_material_interface_contact_loops"
- || opt_key == "support_material_interface_extruder"
- || opt_key == "support_material_interface_spacing"
- || opt_key == "support_material_pattern"
- || opt_key == "support_material_style"
- || opt_key == "support_material_top_interface_pattern"
- || opt_key == "support_material_xy_spacing"
- || opt_key == "support_material_spacing"
- || opt_key == "support_material_closing_radius"
- || opt_key == "support_material_synchronize_layers"
- || opt_key == "support_material_threshold"
- || opt_key == "support_material_with_sheath") {
- steps.emplace_back(posSupportMaterial);
- } else if (opt_key == "bottom_solid_layers") {
- steps.emplace_back(posPrepareInfill);
- if (m_print->config().spiral_vase) {
- // Changing the number of bottom layers when a spiral vase is enabled requires re-slicing the object again.
- // Otherwise, holes in the bottom layers could be filled, as is reported in GH #5528.
- steps.emplace_back(posSlice);
- }
- } else if (
- opt_key == "bottom_solid_min_thickness"
- || opt_key == "ensure_vertical_shell_thickness"
- || opt_key == "interface_shells"
- || opt_key == "infill_extruder"
- || opt_key == "infill_extrusion_change_odd_layers"
- || opt_key == "infill_extrusion_spacing"
- || opt_key == "infill_extrusion_width"
- || opt_key == "infill_every_layers"
- || opt_key == "infill_dense"
- || opt_key == "infill_dense_algo"
- || opt_key == "infill_only_where_needed"
- || opt_key == "ironing"
- || opt_key == "ironing_type"
- || opt_key == "over_bridge_flow_ratio"
- || opt_key == "solid_infill_below_area"
- || opt_key == "solid_infill_below_layer_area"
- || opt_key == "solid_infill_below_width"
- || opt_key == "solid_infill_extruder"
- || opt_key == "solid_infill_every_layers"
- || opt_key == "solid_over_perimeters"
- || opt_key == "top_solid_layers"
- || opt_key == "top_solid_min_thickness") {
- steps.emplace_back(posPrepareInfill);
- } else if (
- opt_key == "bottom_fill_pattern"
- || opt_key == "bridge_fill_pattern"
- || opt_key == "bridge_overlap"
- || opt_key == "bridge_overlap_min"
- || opt_key == "enforce_full_fill_volume"
- || opt_key == "fill_aligned_z"
- || opt_key == "fill_angle"
- || opt_key == "fill_angle_cross"
- || opt_key == "fill_angle_follow_model"
- || opt_key == "fill_angle_increment"
- || opt_key == "fill_angle_template"
- || opt_key == "fill_top_flow_ratio"
- || opt_key == "fill_smooth_width"
- || opt_key == "fill_smooth_distribution"
- || opt_key == "first_layer_infill_extrusion_spacing"
- || opt_key == "first_layer_infill_extrusion_width"
- || opt_key == "infill_anchor"
- || opt_key == "infill_anchor_max"
- || opt_key == "infill_connection"
- || opt_key == "infill_connection_bottom"
- || opt_key == "infill_connection_bridge"
- || opt_key == "infill_connection_solid"
- || opt_key == "infill_connection_top"
- || opt_key == "ironing_angle"
- || opt_key == "ironing_flowrate"
- || opt_key == "ironing_spacing"
- || opt_key == "solid_fill_pattern"
- || opt_key == "top_fill_pattern"
- || opt_key == "top_infill_extrusion_spacing"
- || opt_key == "top_infill_extrusion_width" ) {
- steps.emplace_back(posInfill);
- } else if (opt_key == "fill_pattern") {
- steps.emplace_back(posInfill);
- const auto *old_fill_pattern = old_config.option<ConfigOptionEnum<InfillPattern>>(opt_key);
- const auto *new_fill_pattern = new_config.option<ConfigOptionEnum<InfillPattern>>(opt_key);
- assert(old_fill_pattern && new_fill_pattern);
- // We need to recalculate infill surfaces when infill_only_where_needed is enabled, and we are switching from
- // the Lightning infill to another infill or vice versa.
- //if (m_config.infill_only_where_needed && (new_fill_pattern->value == ipLightning || old_fill_pattern->value == ipLightning))
- //steps.emplace_back(posPrepareInfill);
- } else if (opt_key == "fill_density") {
- // One likely wants to reslice only when switching between zero infill to simulate boolean difference (subtracting volumes),
- // normal infill and 100% (solid) infill.
- const auto *old_density = old_config.option<ConfigOptionPercent>(opt_key);
- const auto *new_density = new_config.option<ConfigOptionPercent>(opt_key);
- assert(old_density && new_density);
- //FIXME Vojtech is not quite sure about the 100% here, maybe it is not needed.
- if (is_approx(old_density->value, 0.) || is_approx(old_density->value, 100.) ||
- is_approx(new_density->value, 0.) || is_approx(new_density->value, 100.)) {
- steps.emplace_back(posPerimeters);
- }
- steps.emplace_back(posPrepareInfill);
- } else if (
- opt_key == "bridge_angle"
- || opt_key == "bridged_infill_margin"
- || opt_key == "extra_perimeters"
- || opt_key == "extra_perimeters_odd_layers"
- || opt_key == "extra_perimeters_on_overhangs"
- || opt_key == "external_infill_margin"
- || opt_key == "external_perimeter_overlap"
- || opt_key == "gap_fill_overlap"
- || opt_key == "infill_overlap"
- || opt_key == "no_perimeter_unsupported_algo"
- || opt_key == "perimeters"
- || opt_key == "perimeters_hole"
- || opt_key == "perimeter_overlap"
- || opt_key == "solid_infill_extrusion_change_odd_layers"
- || opt_key == "solid_infill_extrusion_spacing"
- || opt_key == "solid_infill_extrusion_width"
- || opt_key == "solid_infill_overlap"
- || opt_key == "top_solid_infill_overlap") {
- steps.emplace_back(posPerimeters);
- steps.emplace_back(posPrepareInfill);
- } else if (
- opt_key == "external_perimeter_extrusion_width"
- || opt_key == "external_perimeter_extrusion_spacing"
- || opt_key == "perimeter_extruder"
- || opt_key == "fuzzy_skin"
- || opt_key == "fuzzy_skin_thickness"
- || opt_key == "fuzzy_skin_point_dist"
- || opt_key == "thin_walls") {
- steps.emplace_back(posPerimeters);
- steps.emplace_back(posSupportMaterial);
- } else if (opt_key == "bridge_flow_ratio"
- || opt_key == "extrusion_spacing"
- || opt_key == "extrusion_width"
- || opt_key == "first_layer_extrusion_spacing"
- || opt_key == "first_layer_extrusion_width") {
- //if (m_config.support_material_contact_distance > 0.) {
- // Only invalidate due to bridging if bridging is enabled.
- // If later "support_material_contact_distance" is modified, the complete PrintObject is invalidated anyway.
- steps.emplace_back(posPerimeters);
- steps.emplace_back(posInfill);
- steps.emplace_back(posSupportMaterial);
- //}
- } else if (
- opt_key == "avoid_crossing_top"
- || opt_key == "bridge_acceleration"
- || opt_key == "bridge_speed"
- || opt_key == "brim_acceleration"
- || opt_key == "brim_speed"
- || opt_key == "external_perimeter_speed"
- || opt_key == "default_acceleration"
- || opt_key == "default_speed"
- || opt_key == "external_perimeter_acceleration"
- || opt_key == "external_perimeter_cut_corners"
- || opt_key == "first_layer_acceleration"
- || opt_key == "first_layer_acceleration_over_raft"
- || opt_key == "first_layer_flow_ratio"
- || opt_key == "first_layer_infill_speed"
- || opt_key == "first_layer_min_speed"
- || opt_key == "first_layer_speed"
- || opt_key == "first_layer_speed_over_raft"
- || opt_key == "gap_fill_acceleration"
- || opt_key == "gap_fill_flow_match_perimeter"
- || opt_key == "gap_fill_speed"
- || opt_key == "infill_acceleration"
- || opt_key == "infill_speed"
- || opt_key == "internal_bridge_acceleration"
- || opt_key == "internal_bridge_speed"
- || opt_key == "ironing_acceleration"
- || opt_key == "ironing_speed"
- || opt_key == "milling_after_z"
- || opt_key == "milling_extra_size"
- || opt_key == "milling_post_process"
- || opt_key == "milling_speed"
- || opt_key == "object_gcode"
- || opt_key == "overhangs_acceleration"
- || opt_key == "overhangs_speed"
- || opt_key == "perimeter_acceleration"
- || opt_key == "perimeter_speed"
- || opt_key == "print_extrusion_multiplier"
- || opt_key == "print_first_layer_temperature"
- || opt_key == "print_retract_length"
- || opt_key == "print_retract_lift"
- || opt_key == "print_temperature"
- || opt_key == "region_gcode"
- || opt_key == "seam_position"
- //|| opt_key == "seam_preferred_direction"
- //|| opt_key == "seam_preferred_direction_jitter"
- || opt_key == "seam_angle_cost"
- || opt_key == "seam_notch_all"
- || opt_key == "seam_notch_angle"
- || opt_key == "seam_notch_inner"
- || opt_key == "seam_notch_outer"
- || opt_key == "seam_travel_cost"
- || opt_key == "seam_visibility"
- || opt_key == "small_area_infill_flow_compensation"
- || opt_key == "small_area_infill_flow_compensation_model"
- || opt_key == "small_perimeter_speed"
- || opt_key == "small_perimeter_min_length"
- || opt_key == "small_perimeter_max_length"
- || opt_key == "solid_infill_acceleration"
- || opt_key == "solid_infill_speed"
- || opt_key == "support_material_interface_speed"
- || opt_key == "support_material_speed"
- || opt_key == "thin_walls_acceleration"
- || opt_key == "thin_walls_speed"
- || opt_key == "top_solid_infill_acceleration"
- || opt_key == "top_solid_infill_speed"
- || opt_key == "travel_acceleration"
- || opt_key == "travel_deceleration_use_target") {
- invalidated |= m_print->invalidate_step(psGCodeExport);
- } else if (
- opt_key == "infill_first"
- || opt_key == "wipe_into_infill"
- || opt_key == "wipe_into_objects") {
- invalidated |= m_print->invalidate_step(psWipeTower);
- invalidated |= m_print->invalidate_step(psGCodeExport);
- } else if (
- opt_key == "brim_inside_holes"
- || opt_key == "brim_ears"
- || opt_key == "brim_ears_detection_length"
- || opt_key == "brim_ears_max_angle"
- || opt_key == "brim_ears_pattern"
- || opt_key == "brim_per_object"
- || opt_key == "brim_separation"
- || opt_key == "brim_type") {
- invalidated |= m_print->invalidate_step(psSkirtBrim);
- // Brim is printed below supports, support invalidates brim and skirt.
- steps.emplace_back(posSupportMaterial);
- } else if (
- opt_key == "brim_width"
- || opt_key == "brim_width_interior") {
- invalidated |= m_print->invalidate_step(psSkirtBrim);
- // these two may change the ordering of first layer perimeters
- steps.emplace_back(posPerimeters);
- // Brim is printed below supports, support invalidates brim and skirt.
- steps.emplace_back(posSupportMaterial);
- } else {
- // for legacy, if we can't handle this option let's invalidate all steps
- this->invalidate_all_steps();
- invalidated = true;
- }
- }
- sort_remove_duplicates(steps);
- for (PrintObjectStep step : steps)
- invalidated |= this->invalidate_step(step);
- return invalidated;
- }
- bool PrintObject::invalidate_step(PrintObjectStep step)
- {
- bool invalidated = Inherited::invalidate_step(step);
-
- // propagate to dependent steps
- if (step == posPerimeters) {
- invalidated |= this->invalidate_steps({ posPrepareInfill, posInfill, posIroning,
- posSupportSpotsSearch, posEstimateCurledExtrusions, posCalculateOverhangingPerimeters, posSimplifyPath });
- invalidated |= m_print->invalidate_steps({ psSkirtBrim });
- } else if (step == posPrepareInfill) {
- invalidated |= this->invalidate_steps({ posInfill, posIroning, posSupportSpotsSearch, posSimplifyPath });
- } else if (step == posInfill) {
- invalidated |= this->invalidate_steps({ posIroning, posSupportSpotsSearch, posSimplifyPath });
- invalidated |= m_print->invalidate_steps({ psSkirtBrim });
- } else if (step == posSlice) {
- invalidated |= this->invalidate_steps({posPerimeters, posPrepareInfill, posInfill, posIroning, posSupportSpotsSearch,
- posSupportMaterial, posEstimateCurledExtrusions, posCalculateOverhangingPerimeters,
- posSimplifyPath });
- invalidated |= m_print->invalidate_steps({ psSkirtBrim });
- m_slicing_params->valid = false;
- } else if (step == posSupportMaterial) {
- invalidated |= m_print->invalidate_steps({ psSkirtBrim, });
- invalidated |= this->invalidate_steps({ posEstimateCurledExtrusions });
- m_slicing_params->valid = false;
- }
- // invalidate alerts step always, since it depends on everything (except supports, but with supports enabled it is skipped anyway.)
- invalidated |= m_print->invalidate_step(psAlertWhenSupportsNeeded);
- // Wipe tower depends on the ordering of extruders, which in turn depends on everything.
- // It also decides about what the wipe_into_infill / wipe_into_object features will do,
- // and that too depends on many of the settings.
- invalidated |= m_print->invalidate_step(psWipeTower);
- // Invalidate G-code export in any case.
- invalidated |= m_print->invalidate_step(psGCodeExport);
- return invalidated;
- }
- bool PrintObject::invalidate_all_steps()
- {
- // First call the "invalidate" functions, which may cancel background processing.
- bool result = Inherited::invalidate_all_steps() | m_print->invalidate_all_steps();
- // Then reset some of the depending values.
- m_slicing_params->valid = false;
- return result;
- }
- // Called on main thread with stopped or paused background processing to let PrintObject release data for its milestones that were invalidated or canceled.
- void PrintObject::cleanup()
- {
- if (this->query_reset_dirty_step_unguarded(posInfill))
- this->clear_fills();
- if (this->query_reset_dirty_step_unguarded(posSupportMaterial))
- this->clear_support_layers();
- }
- //Fit to size helper
- bool has_boundary_point(const MultiPoint& contour, const Point &point)
- {
- double dist = (contour.point_projection(point).first - point).cast<double>().norm();
- return dist < SCALED_EPSILON;
- }
- bool has_boundary_point(const ExPolygon& expoly, const Point &point)
- {
- if (has_boundary_point(expoly.contour, point)) return true;
- for (Polygons::const_iterator h = expoly.holes.begin(); h != expoly.holes.end(); ++h) {
- if (has_boundary_point(*h,point)) return true;
- }
- return false;
- }
- // Function used by fit_to_size.
- // It check if polygon_to_check can be decimated, using only point into allowedPoints and also cover polygon_to_cover
- ExPolygon try_fit_to_size(ExPolygon polygon_to_check, const ExPolygons& allowedPoints) {
-
- ExPolygon polygon_reduced = polygon_to_check;
- size_t pos_check = 0;
- bool has_del = false;
- while ((polygon_reduced.contour.points.begin() + pos_check) != polygon_reduced.contour.points.end()) {
- bool ok = false;
- for (const ExPolygon &poly : allowedPoints) {
- //return this->contains(point) || this->has_boundary_point(point);
- if (poly.contains(*(polygon_reduced.contour.points.begin() + pos_check))
- || has_boundary_point(poly, *(polygon_reduced.contour.points.begin() + pos_check))) {
- ok = true;
- has_del = true;
- break;
- }
- }
- if (ok) ++pos_check;
- else polygon_reduced.contour.points.erase(polygon_reduced.contour.points.begin() + pos_check);
- }
- if (has_del) polygon_reduced.holes.clear();
- return polygon_reduced;
- }
- ExPolygon try_fit_to_size2(ExPolygon polygon_to_check, const ExPolygon& allowedPoints) {
- ExPolygon polygon_reduced = polygon_to_check;
- size_t pos_check = 0;
- while ((polygon_reduced.contour.points.begin() + pos_check) != polygon_reduced.contour.points.end()) {
- //Point best_point = polygon_reduced.contour.points[pos_check].projection_onto(allowedPoints.contour);
- Point best_point = allowedPoints.contour.point_projection(polygon_reduced.contour.points[pos_check]).first;
- for (const Polygon& hole : allowedPoints.holes) {
- Point hole_point = hole.point_projection(polygon_reduced.contour.points[pos_check]).first;
- if ((hole_point - polygon_reduced.contour.points[pos_check]).norm() < (best_point - polygon_reduced.contour.points[pos_check]).norm())
- best_point = hole_point;
- }
- if ((best_point - polygon_reduced.contour.points[pos_check]).norm() < scale_(0.01)) ++pos_check;
- else polygon_reduced.contour.points.erase(polygon_reduced.contour.points.begin() + pos_check);
- }
- polygon_reduced.holes.clear();
- return polygon_reduced;
- }
- // find one of the smallest polygon, growing polygon_to_cover, only using point into growing_area and covering polygon_to_cover.
- ExPolygons dense_fill_fit_to_size(const ExPolygon& bad_polygon_to_cover,
- const ExPolygon& growing_area, const coord_t offset, float coverage) {
- //fix uncoverable area
- ExPolygons polygons_to_cover = intersection_ex(bad_polygon_to_cover, growing_area);
- if (polygons_to_cover.size() != 1)
- return { growing_area };
- const ExPolygon polygon_to_cover = polygons_to_cover.front();
- //grow the polygon_to_check enough to cover polygon_to_cover
- float current_coverage = coverage;
- coord_t previous_offset = 0;
- coord_t current_offset = offset;
- ExPolygon polygon_reduced = try_fit_to_size2(polygon_to_cover, growing_area);
- while (polygon_reduced.empty()) {
- current_offset *= 2;
- ExPolygons bigger_polygon = offset_ex(polygon_to_cover, double(current_offset));
- if (bigger_polygon.size() != 1) break;
- bigger_polygon = intersection_ex(bigger_polygon[0], growing_area);
- if (bigger_polygon.size() != 1) break;
- polygon_reduced = try_fit_to_size2(bigger_polygon[0], growing_area);
- }
- //ExPolygons to_check = offset_ex(polygon_to_cover, -offset);
- ExPolygons not_covered = diff_ex(polygon_to_cover, polygon_reduced, ApplySafetyOffset::Yes);
- while (!not_covered.empty()) {
- //not enough, use a bigger offset
- float percent_coverage = (float)(polygon_reduced.area() / growing_area.area());
- float next_coverage = percent_coverage + (percent_coverage - current_coverage) * 4;
- previous_offset = current_offset;
- current_offset *= 2;
- if (next_coverage < 0.1) current_offset *= 2;
- //create the bigger polygon and test it
- ExPolygons bigger_polygon = offset_ex(polygon_to_cover, double(current_offset));
- if (bigger_polygon.size() != 1) {
- // Error, growing a single polygon result in many/no other => abord
- return ExPolygons();
- }
- bigger_polygon = intersection_ex(bigger_polygon[0], growing_area);
- // After he intersection, we may have section of the bigger_polygon that jumped over a 'clif' to exist in an other area, have to remove them.
- if (bigger_polygon.size() > 1) {
- //remove polygon not in intersection with polygon_to_cover
- for (int i = 0; i < (int)bigger_polygon.size(); i++) {
- if (intersection_ex(bigger_polygon[i], polygon_to_cover).empty()) {
- bigger_polygon.erase(bigger_polygon.begin() + i);
- i--;
- }
- }
- }
- if (bigger_polygon.size() != 1 || bigger_polygon[0].area() > growing_area.area()) {
- // Growing too much => we can as well use the full coverage, in this case
- polygon_reduced = growing_area;
- break;
- //return ExPolygons() = { growing_area };
- }
- //polygon_reduced = try_fit_to_size(bigger_polygon[0], allowedPoints);
- polygon_reduced = try_fit_to_size2(bigger_polygon[0], growing_area);
- not_covered = diff_ex(polygon_to_cover, polygon_reduced, ApplySafetyOffset::Yes);
- }
- //ok, we have a good one, now try to optimise (unless there are almost no growth)
- if (current_offset > offset * 3) {
- //try to shrink
- uint32_t nb_opti_max = 6;
- for (uint32_t i = 0; i < nb_opti_max; ++i) {
- coord_t new_offset = (previous_offset + current_offset) / 2;
- ExPolygons bigger_polygon = offset_ex(polygon_to_cover, double(new_offset));
- if (bigger_polygon.size() != 1) {
- //Warn, growing a single polygon result in many/no other, use previous good result
- break;
- }
- bigger_polygon = intersection_ex(bigger_polygon[0], growing_area);
- if (bigger_polygon.size() != 1 || bigger_polygon[0].area() > growing_area.area()) {
- //growing too much, use previous good result (imo, should not be possible to enter this branch)
- break;
- }
- //ExPolygon polygon_test = try_fit_to_size(bigger_polygon[0], allowedPoints);
- ExPolygon polygon_test = try_fit_to_size2(bigger_polygon[0], growing_area);
- not_covered = diff_ex(polygon_to_cover, polygon_test, ApplySafetyOffset::Yes);
- if (!not_covered.empty()) {
- //bad, not enough, use a bigger offset
- previous_offset = new_offset;
- } else {
- //good, we may now try a smaller offset
- current_offset = new_offset;
- polygon_reduced = polygon_test;
- }
- }
- }
- //return the area which cover the growing_area. Intersect it to retreive the holes.
- ExPolygons to_print = intersection_ex(polygon_reduced, growing_area);
- //remove polygon not in intersection with polygon_to_cover
- for (int i = 0; i < (int)to_print.size(); i++) {
- if (intersection_ex(to_print[i], polygon_to_cover).empty()) {
- to_print.erase(to_print.begin() + i);
- i--;
- }
- }
- return to_print;
- }
- void PrintObject::tag_under_bridge() {
- const float COEFF_SPLIT = 1.5;
- coord_t scaled_resolution = std::max(SCALED_EPSILON, scale_t(this->print()->config().resolution.value));
- for (size_t region_idx = 0; region_idx < this->print()->num_print_regions(); ++ region_idx) {
- const PrintRegion* region = &this->print()->get_print_region(region_idx);
- //count how many surface there are on each one
- if (region->config().infill_dense.get_bool() && region->config().fill_density < 40) {
- std::vector<LayerRegion*> layeridx2lregion;
- std::vector<Surfaces> new_surfaces; //surface store, as you can't modify them when working in //
- // store the LayerRegion on which we are working
- layeridx2lregion.resize(this->layers().size(), nullptr);
- new_surfaces.resize(this->layers().size(), Surfaces{});
- for (size_t idx_layer = 0; idx_layer < this->layers().size(); ++idx_layer) {
- LayerRegion* layerm = nullptr;
- for (LayerRegion* lregion : this->layers()[idx_layer]->regions()) {
- if (&lregion->region() == region) {
- layerm = lregion;
- break;
- }
- }
- if (layerm != nullptr)
- layeridx2lregion[idx_layer] = layerm;
- }
- // run in parallel, it's a costly thing.
- Slic3r::parallel_for(size_t(0), this->layers().size() - 1,
- [this, &layeridx2lregion, &new_surfaces, region, COEFF_SPLIT, scaled_resolution](const size_t idx_layer) {
- // we our LayerRegion and the one on top
- LayerRegion* layerm = layeridx2lregion[idx_layer];
- const LayerRegion* previousOne = nullptr;
- previousOne = layeridx2lregion[idx_layer + 1];
- if (layerm != nullptr && previousOne != nullptr) {
- Surfaces &surfs_to_add = new_surfaces[idx_layer];
- // check all surfaces to cover
- for (Surface& surface : layerm->set_fill_surfaces().surfaces) {
- surface.maxNbSolidLayersOnTop = -1;
- if (!surface.has_fill_solid()) {
- Surfaces surf_to_add;
- ExPolygons dense_polys;
- std::vector<uint16_t> dense_priority;
- const ExPolygons surfs_with_overlap = { surface.expolygon };
- // create a surface with overlap to allow the dense thing to bond to the infill
- coord_t scaled_width = layerm->flow(frInfill).scaled_width();
- coord_t overlap = scaled_width / 4;
- for (const ExPolygon& surf_with_overlap : surfs_with_overlap) {
- ExPolygons sparse_polys = { surf_with_overlap };
- //find the surface which intersect with the smallest maxNb possible
- for (const Surface& upp : previousOne->fill_surfaces().surfaces) {
- if (upp.has_fill_solid()) {
- // i'm using intersection_ex because the result different than
- // upp.expolygon.overlaps(surf.expolygon) or surf.expolygon.overlaps(upp.expolygon)
- // and a little offset2 to remove the almost supported area
- ExPolygons intersect =
- offset2_ex(
- intersection_ex(sparse_polys, ExPolygons{ upp.expolygon }, ApplySafetyOffset::Yes)
- , (float)-layerm->flow(frInfill).scaled_width(), (float)layerm->flow(frInfill).scaled_width());
- if (!intersect.empty()) {
- DenseInfillAlgo algo = layerm->region().config().infill_dense_algo.value;
- //if no infill, don't bother, it's always yes
- if (region->config().fill_density.value == 0) {
- if (dfaAutoOrEnlarged == algo)
- algo = dfaAutomatic;
- else if (dfaAutomatic != algo)
- algo = dfaAutoNotFull;
- }
- if ( dfaAutoOrNothing == algo
- || dfaAutoOrEnlarged == algo) {
- //check if small enough
- double max_nozzle_diam = 0;
- for (uint16_t extruder_id : object_extruders()) {
- max_nozzle_diam = std::max(max_nozzle_diam, print()->config().nozzle_diameter.get_at(extruder_id));
- }
- coordf_t min_width = scale_d(max_nozzle_diam) / region->config().fill_density.get_abs_value(1.);
- ExPolygons smalls = offset_ex(intersect, -min_width);
- //small enough ?
- if (smalls.empty()) {
- if (dfaAutoOrNothing == algo)
- algo = dfaAutoNotFull;
- if (dfaAutoOrEnlarged == algo)
- algo = dfaAutomatic;
- } else if (dfaAutoOrNothing == algo) {
- algo = dfaDisabled;
- }
- }
- if (dfaEnlarged == algo) {
- //expand the area a bit
- intersect = offset_ex(intersect, (scaled(layerm->region().config().external_infill_margin.get_abs_value(
- region->config().perimeters == 0 ? 0 : (layerm->flow(frExternalPerimeter).width() + layerm->flow(frPerimeter).spacing() * (region->config().perimeters - 1))))));
- intersect = intersection_ex(intersect, sparse_polys);
- } else if (dfaDisabled == algo) {
- intersect.clear();
- } else {
- double sparse_area = surf_with_overlap.area();
- double area_to_cover = 0;
- if (dfaAutoNotFull == algo) {
- // calculate area to decide if area is small enough for autofill
- for (ExPolygon poly_inter : intersect)
- area_to_cover += poly_inter.area();
- // if we have to fill everything, don't bother
- if (area_to_cover * 1.1 > sparse_area)
- intersect.clear();
- }
- //like intersect.empty() but more resilient
- ExPolygons cover_intersect;
- // it will be a dense infill, split the surface if needed
- //ExPolygons cover_intersect;
- for (ExPolygon& expoly_tocover : intersect) {
- ExPolygons temp = dense_fill_fit_to_size(
- expoly_tocover,
- surf_with_overlap,
- 4 * layerm->flow(frInfill).scaled_width(),
- 0.01f);
- cover_intersect.insert(cover_intersect.end(), temp.begin(), temp.end());
- }
- // calculate area to decide if area is small enough for autofill
- if (dfaAutoOrEnlarged == algo) {
- double area_dense_covered = 0;
- for (ExPolygon poly_inter : cover_intersect)
- area_dense_covered += poly_inter.area();
- // if enlarge is smaller, use enlarge
- intersect = offset_ex(intersect, (scaled(layerm->region().config().external_infill_margin.get_abs_value(
- region->config().perimeters == 0 ? 0 : (layerm->flow(frExternalPerimeter).width() + layerm->flow(frPerimeter).spacing() * (region->config().perimeters - 1))))));
- intersect = intersection_ex(intersect, sparse_polys);
- double area_enlarged_covered = 0;
- for (ExPolygon poly_inter : intersect)
- area_enlarged_covered += poly_inter.area();
- if (area_dense_covered < area_enlarged_covered) {
- intersect = cover_intersect;
- }
- }else
- intersect = cover_intersect;
- }
- if (!intersect.empty()) {
- ExPolygons sparse_surfaces = diff_ex(sparse_polys, intersect, ApplySafetyOffset::Yes);
- ExPolygons dense_surfaces = diff_ex(sparse_polys, sparse_surfaces, ApplySafetyOffset::Yes);
- for (ExPolygon& poly : intersect) {
- uint16_t priority = 1;
- ExPolygons dense = { poly };
- for (size_t idx_dense = 0; idx_dense < dense_polys.size(); idx_dense++) {
- ExPolygons dense_test = diff_ex(dense, ExPolygons{ dense_polys[idx_dense] }, ApplySafetyOffset::Yes);
- if (dense_test != dense) {
- priority = std::max(priority, uint16_t(dense_priority[idx_dense] + 1));
- }
- dense = dense_test;
- }
- dense_polys.insert(dense_polys.end(), dense.begin(), dense.end());
- for (size_t i = 0; i < dense.size(); i++)
- dense_priority.push_back(priority);
- }
- //assign (copy)
- sparse_polys = std::move(sparse_surfaces);
- }
- }
- }
- //check if we are full-dense
- if (sparse_polys.empty()) break;
- }
- //check if we need to split the surface
- if (!dense_polys.empty()) {
- double area_dense = 0;
- for (ExPolygon poly_inter : dense_polys) area_dense += poly_inter.area();
- double area_sparse = 0;
- for (ExPolygon poly_inter : sparse_polys) area_sparse += poly_inter.area();
- // if almost no empty space, simplify by filling everything (else)
- if (area_sparse > area_dense * 0.1) {
- //split
- //dense_polys = union_ex(dense_polys);
- for (size_t idx_dense = 0; idx_dense < dense_polys.size(); idx_dense++) {
- ExPolygon dense_poly = dense_polys[idx_dense];
- //remove overlap with perimeter
- ExPolygons offseted_dense_polys = layerm->fill_no_overlap_expolygons().empty()
- ? ExPolygons{dense_poly}
- : intersection_ex(ExPolygons{ dense_poly }, layerm->fill_no_overlap_expolygons());
- //add overlap with everything
- offseted_dense_polys = offset_ex(offseted_dense_polys, overlap);
- ensure_valid(offseted_dense_polys, scaled_resolution);
- for (ExPolygon offseted_dense_poly : offseted_dense_polys) {
- Surface dense_surf(surface, offseted_dense_poly);
- dense_surf.maxNbSolidLayersOnTop = 1;
- dense_surf.priority = dense_priority[idx_dense];
- surf_to_add.push_back(dense_surf);
- }
- }
- sparse_polys = union_ex(sparse_polys);
- ensure_valid(sparse_polys, scaled_resolution);
- for (ExPolygon sparse_poly : sparse_polys) {
- Surface sparse_surf(surface, sparse_poly);
- surf_to_add.push_back(sparse_surf);
- }
- //layerm->fill_surfaces.surfaces.erase(it_surf);
- } else {
- surface.maxNbSolidLayersOnTop = 1;
- surf_to_add.clear();
- surface.expolygon.assert_valid();
- surf_to_add.push_back(surface);
- break;
- }
- } else {
- surf_to_add.clear();
- surface.expolygon.assert_valid();
- surf_to_add.emplace_back(std::move(surface));
- // mitigation: if not possible, don't try the others.
- break;
- }
- }
- // break go here
- for(Surface &srf : surf_to_add) srf.expolygon.assert_valid();
- surfs_to_add.insert(surfs_to_add.begin(), surf_to_add.begin(), surf_to_add.end());
- } else {
- surface.expolygon.assert_valid();
- surfs_to_add.emplace_back(std::move(surface));
- }
- }
- //layerm->fill_surfaces.surfaces = std::move(surfs_to_add);
- }
- });
- // now set the new surfaces
- for (size_t idx_layer = 0; idx_layer < this->layers().size() - 1; ++idx_layer) {
- LayerRegion* lr = layeridx2lregion[idx_layer];
- if(lr != nullptr && layeridx2lregion[idx_layer + 1] != nullptr) {
- for(size_t i = 0; i < new_surfaces[idx_layer].size(); ++i) {
- new_surfaces[idx_layer][i].expolygon.assert_valid();
- if(new_surfaces[idx_layer][i].expolygon.contour.size() < 3){
- new_surfaces[idx_layer].erase(new_surfaces[idx_layer].begin() + i);
- --i;
- }
- }
- lr->set_fill_surfaces().surfaces = new_surfaces[idx_layer];
- }
- }
- }
- }
- }
- // This function analyzes slices of a region (SurfaceCollection slices).
- // Each region slice (instance of Surface) is analyzed, whether it is supported or whether it is the top surface.
- // Initially all slices are of type stInternal.
- // Slices are compared against the top / bottom slices and regions and classified to the following groups:
- // stTop - Part of a region, which is not covered by any upper layer. This surface will be filled with a top solid infill.
- // stBottomBridge - Part of a region, which is not fully supported, but it hangs in the air, or it hangs losely on a support or a raft.
- // stBottom - Part of a region, which is not supported by the same region, but it is supported either by another region, or by a soluble interface layer.
- // stInternal - Part of a region, which is supported by the same region type.
- // If a part of a region is of stBottom and stTop, the stBottom wins.
- void PrintObject::detect_surfaces_type()
- {
- BOOST_LOG_TRIVIAL(info) << "Detecting solid surfaces..." << log_memory_info();
- // Interface shells: the intersecting parts are treated as self standing objects supporting each other.
- // Each of the objects will have a full number of top / bottom layers, even if these top / bottom layers
- // are completely hidden inside a collective body of intersecting parts.
- // This is useful if one of the parts is to be dissolved, or if it is transparent and the internal shells
- // should be visible.
- bool spiral_vase = this->print()->config().spiral_vase.value;
- bool interface_shells = ! spiral_vase && m_config.interface_shells.value;
- size_t num_layers = spiral_vase ? std::min(size_t(this->printing_region(0).config().bottom_solid_layers), m_layers.size()) : m_layers.size();
- for (size_t region_id = 0; region_id < this->num_printing_regions(); ++ region_id) {
- BOOST_LOG_TRIVIAL(debug) << "Detecting solid surfaces for region " << region_id << " in parallel - start";
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- for (Layer *layer : m_layers)
- layer->m_regions[region_id]->export_region_fill_surfaces_to_svg_debug("1_detect_surfaces_type-initial");
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- // If interface shells are allowed, the region->surfaces cannot be overwritten as they may be used by other threads.
- // Cache the result of the following parallel_loop.
- std::vector<Surfaces> surfaces_new;
- if (interface_shells)
- surfaces_new.assign(num_layers, Surfaces());
- // If we have soluble support material, don't bridge. The overhang will be squished against a soluble layer separating
- // the support from the print.
- bool has_bridges = !(m_config.support_material.value
- && m_config.support_material_contact_distance_type.value == zdNone
- && !m_config.dont_support_bridges);
- SurfaceType surface_type_bottom_other =
- !has_bridges ?
- stPosBottom | stDensSolid :
- stPosBottom | stDensSolid | stModBridge;
-
- coord_t scaled_resolution = std::max(SCALED_EPSILON, scale_t(print()->config().resolution.value));
- Slic3r::parallel_for(size_t(0),
- spiral_vase ?
- // In spiral vase mode, reserve the last layer for the top surface if more than 1 layer is planned for the vase bottom.
- ((num_layers > 1) ? num_layers - 1 : num_layers) :
- // In non-spiral vase mode, go over all layers.
- m_layers.size(),
- [this, region_id, interface_shells, &surfaces_new, has_bridges, surface_type_bottom_other, scaled_resolution]
- (const size_t idx_layer) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- m_print->throw_if_canceled();
- // BOOST_LOG_TRIVIAL(trace) << "Detecting solid surfaces for region " << region_id << " and layer " << layer->print_z;
- Layer *layer = m_layers[idx_layer];
- LayerRegion *layerm = layer->get_region(region_id);
- // comparison happens against the *full* slices (considering all regions)
- // unless internal shells are requested
- Layer *upper_layer = (idx_layer + 1 < this->layer_count()) ? m_layers[idx_layer + 1] : nullptr;
- Layer *lower_layer = (idx_layer > 0) ? m_layers[idx_layer - 1] : nullptr;
- // collapse very narrow parts (using the safety offset in the diff is not enough)
- float offset = layerm->flow(frExternalPerimeter).scaled_width() / 10.f;
- ExPolygons layerm_slices_surfaces = to_expolygons(layerm->slices().surfaces);
- // no_perimeter_full_bridge allow to put bridges where there are nothing, hence adding area to slice, that's why we need to start from the result of PerimeterGenerator.
- if (layerm->region().config().no_perimeter_unsupported_algo.value == npuaFilled) {
- append(layerm_slices_surfaces, to_expolygons(layerm->fill_surfaces().surfaces));
- layerm_slices_surfaces = union_ex(layerm_slices_surfaces);
- assert_valid(layerm_slices_surfaces);
- }
- // find top surfaces (difference between current surfaces
- // of current layer and upper one)
- Surfaces top;
- if (upper_layer) {
- for(auto &srf : top) srf.expolygon.assert_valid();
- ExPolygons upper_slices = interface_shells ?
- diff_ex(layerm_slices_surfaces, upper_layer->get_region(region_id)->slices().surfaces, ApplySafetyOffset::Yes) :
- diff_ex(layerm_slices_surfaces, upper_layer->lslices(), ApplySafetyOffset::Yes);
- ExPolygons new_top_surfaces = opening_ex(upper_slices, offset);
- ensure_valid(new_top_surfaces, scaled_resolution);
- assert_valid(new_top_surfaces);
- surfaces_append(top, std::move(new_top_surfaces), stPosTop | stDensSolid);
- for(Surface &srf : top) srf.expolygon.assert_valid();
- } else {
- // if no upper layer, all surfaces of this one are solid
- // we clone surfaces because we're going to clear the slices collection
- top = layerm->slices().surfaces;
- for (Surface &surface : top)
- surface.surface_type = stPosTop | stDensSolid;
- for(Surface &srf : top) srf.expolygon.assert_valid();
- }
-
- // Find bottom surfaces (difference between current surfaces of current layer and lower one).
- Surfaces bottom;
- if (lower_layer) {
- #if 0
- //FIXME Why is this branch failing t\multi.t ?
- Polygons lower_slices = interface_shells ?
- to_polygons(lower_layer->get_region(region_id)->slices.surfaces) :
- to_polygons(lower_layer->slices);
- surfaces_append(bottom,
- opening_ex(diff(layerm_slices_surfaces, lower_slices, true), offset),
- surface_type_bottom_other);
- #else
- // Any surface lying on the void is a true bottom bridge (an overhang)
- ExPolygons new_bot_surfs = opening_ex(
- diff_ex(layerm_slices_surfaces, lower_layer->lslices(), ApplySafetyOffset::Yes),
- offset);
- ensure_valid(new_bot_surfs, scaled_resolution);
- assert_valid(new_bot_surfs);
- for(Surface &srf : bottom) srf.expolygon.assert_valid();
- surfaces_append(
- bottom,
- std::move(new_bot_surfs),
- surface_type_bottom_other);
- for(auto &srf : bottom) srf.expolygon.assert_valid();
- // if user requested internal shells, we need to identify surfaces
- // lying on other slices not belonging to this region
- if (interface_shells) {
- // non-bridging bottom surfaces: any part of this layer lying
- // on something else, excluding those lying on our own region
- ExPolygons new_bot_interface_surfs =
- opening_ex(diff_ex(intersection(layerm_slices_surfaces,
- lower_layer->lslices()), // supported
- lower_layer->get_region(region_id)->slices().surfaces,
- ApplySafetyOffset::Yes),
- offset); //-+
- ensure_valid(new_bot_interface_surfs, scaled_resolution);
- surfaces_append(bottom, std::move(new_bot_interface_surfs), stPosBottom | stDensSolid);
- for(Surface &srf : bottom) srf.expolygon.assert_valid();
- }
- #endif
- } else {
- // if no lower layer, all surfaces of this one are solid
- // we clone surfaces because we're going to clear the slices collection
- bottom = layerm->slices().surfaces;
- // Note: PS 2.4 changed that by "no bridge"... i dont know why?
- for (Surface& surface : bottom)
- surface.surface_type = //stPosBottom | stDensSolid;
- (m_config.raft_layers.value > 0 && m_config.support_material_contact_distance_type.value != zdNone) ?
- stPosBottom | stDensSolid | stModBridge : stPosBottom | stDensSolid;
- }
-
- // now, if the object contained a thin membrane, we could have overlapping bottom
- // and top surfaces; let's do an intersection to discover them and consider them
- // as bottom surfaces (to allow for bridge detection)
- if (! top.empty() && ! bottom.empty()) {
- // Polygons overlapping = intersection(to_polygons(top), to_polygons(bottom));
- // Slic3r::debugf " layer %d contains %d membrane(s)\n", $layerm->layer->id, scalar(@$overlapping)
- // if $Slic3r::debug;
- Polygons top_polygons = to_polygons(std::move(top));
- assert_valid(top_polygons);
- for(auto &srf : bottom) srf.expolygon.assert_valid();
- top.clear();
- ExPolygons diff = diff_ex(top_polygons, bottom);
- ensure_valid(diff, scaled_resolution);
- assert_valid(diff);
- surfaces_append(top, std::move(diff), stPosTop | stDensSolid);
- }
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- {
- static int iRun = 0;
- std::vector<std::pair<Slic3r::ExPolygons, SVG::ExPolygonAttributes>> expolygons_with_attributes;
- expolygons_with_attributes.emplace_back(std::make_pair(union_ex(top), SVG::ExPolygonAttributes("green")));
- expolygons_with_attributes.emplace_back(std::make_pair(union_ex(bottom), SVG::ExPolygonAttributes("brown")));
- expolygons_with_attributes.emplace_back(std::make_pair(to_expolygons(layerm->slices().surfaces), SVG::ExPolygonAttributes("black")));
- SVG::export_expolygons(debug_out_path("1_detect_surfaces_type_%d_region%d-layer_%f.svg", iRun ++, region_id, layer->print_z).c_str(), expolygons_with_attributes);
- }
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- // save surfaces to layer
- Surfaces &surfaces_out = interface_shells ? surfaces_new[idx_layer] : layerm->m_slices.surfaces;
- Surfaces surfaces_backup;
- if (! interface_shells) {
- surfaces_backup = std::move(surfaces_out);
- for(auto &srf : surfaces_backup) srf.expolygon.assert_valid();
- surfaces_out.clear();
- }
- //const Surfaces &surfaces_prev = interface_shells ? layerm_slices_surfaces : surfaces_backup;
- const ExPolygons& surfaces_prev_expolys = interface_shells ? layerm_slices_surfaces : to_expolygons(surfaces_backup);
- // find internal surfaces (difference between top/bottom surfaces and others)
- {
- Polygons topbottom = to_polygons(top);
- polygons_append(topbottom, to_polygons(bottom));
- assert_valid(topbottom);
- assert_valid(surfaces_prev_expolys);
- ExPolygons diff = diff_ex(surfaces_prev_expolys, topbottom);
- ensure_valid(diff, scaled_resolution);
- assert_valid(diff);
- surfaces_append(surfaces_out, std::move(diff), stPosInternal | stDensSparse);
- }
-
- for(Surface &srf : top) srf.expolygon.assert_valid();
- for(Surface &srf : bottom) srf.expolygon.assert_valid();
- surfaces_append(surfaces_out, std::move(top));
- surfaces_append(surfaces_out, std::move(bottom));
-
- // Slic3r::debugf " layer %d has %d bottom, %d top and %d internal surfaces\n",
- // $layerm->layer->id, scalar(@bottom), scalar(@top), scalar(@internal) if $Slic3r::debug;
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- layerm->export_region_slices_to_svg_debug("detect_surfaces_type-final");
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- }
- ); // for each layer of a region
- m_print->throw_if_canceled();
- if (interface_shells) {
- // Move surfaces_new to layerm->slices.surfaces
- // not '< num_layers' because spiral vase don't need it?
- for (size_t idx_layer = 0; idx_layer < m_layers.size(); ++idx_layer)
- m_layers[idx_layer]->get_region(region_id)->m_slices.surfaces = std::move(surfaces_new[idx_layer]);
- }
- if (spiral_vase) {
- if (num_layers > 1)
- // Turn the last bottom layer infill to a top infill, so it will be extruded with a proper pattern.
- m_layers[num_layers - 1]->m_regions[region_id]->m_slices.set_type((stPosTop | stDensSolid));
- for (size_t i = num_layers; i < m_layers.size(); ++i)
- m_layers[i]->m_regions[region_id]->m_slices.set_type((stPosInternal | stDensSparse));
- }
- BOOST_LOG_TRIVIAL(debug) << "Detecting solid surfaces for region " << region_id << " - clipping in parallel - start";
- // Fill in layerm->fill_surfaces by trimming the layerm->slices by the cummulative layerm->fill_surfaces.
- Slic3r::parallel_for(size_t(0), m_layers.size(),
- [this, region_id](const size_t idx_layer) {
- m_print->throw_if_canceled();
- LayerRegion* layerm = m_layers[idx_layer]->get_region(region_id);
- layerm->slices_to_fill_surfaces_clipped(
- std::max(SCALED_EPSILON * 2,
- std::max(scale_t(m_print->config().resolution) / 4,
- scale_t(m_print->config().resolution_internal) / 8)));
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- layerm->export_region_fill_surfaces_to_svg_debug("1_detect_surfaces_type-final");
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- } // for each layer of a region
- );
- m_print->throw_if_canceled();
- BOOST_LOG_TRIVIAL(debug) << "Detecting solid surfaces for region " << region_id << " - clipping in parallel - end";
- } // for each this->print->region_count
- // Mark the object to have the region slices classified (typed, which also means they are split based on whether they are supported, bridging, top layers etc.)
- m_typed_slices = true;
- }
-
- void PrintObject::apply_solid_infill_below_layer_area()
- {
- // compute the total layer surface for the bed, for solid_infill_below_layer_area
- for (auto *my_layer : this->m_layers) {
- bool exists = false;
- for (auto *region : my_layer->m_regions) {
- exists |= region->region().config().solid_infill_below_layer_area.value > 0;
- }
- if (!exists)
- return;
- double total_area = 0;
- if (this->print()->config().complete_objects.value) {
- // sequential printing: only consider myself
- for (const ExPolygon &slice : my_layer->lslices()) { total_area += slice.area(); }
- } else {
- // parallel printing: get all objects
- for (const PrintObject *object : this->print()->objects()) {
- for (auto *layer : object->m_layers) {
- if (std::abs(layer->print_z - my_layer->print_z) < EPSILON) {
- for (const ExPolygon &slice : layer->lslices()) { total_area += slice.area(); }
- }
- }
- }
- }
- // is it low enough to apply solid_infill_below_layer_area?
- for (auto *region : my_layer->m_regions) {
- if (!this->print()->config().spiral_vase.value && region->region().config().fill_density.value > 0) {
- double min_area = scale_d(scale_d(region->region().config().solid_infill_below_layer_area.value));
- for (Surface& surface : region->set_fill_surfaces()) {
- if (surface.has_fill_sparse() && surface.has_pos_internal() && total_area <= min_area)
- surface.surface_type = stPosInternal | stDensSolid;
- }
- }
- }
- }
- }
- void PrintObject::process_external_surfaces()
- {
- BOOST_LOG_TRIVIAL(info) << "Processing external surfaces..." << log_memory_info();
- // Cached surfaces covered by some extrusion, defining regions, over which the from the surfaces one layer higher are allowed to expand.
- std::vector<Polygons> surfaces_covered;
- // Is there any printing region, that has zero infill? If so, then we don't want the expansion to be performed over the complete voids, but only
- // over voids, which are supported by the layer below.
- bool has_voids = false;
- for (size_t region_id = 0; region_id < this->num_printing_regions(); ++ region_id)
- if (this->printing_region(region_id).config().fill_density == 0) {
- has_voids = true;
- break;
- }
- if (has_voids && m_layers.size() > 1) {
- // All but stInternal-sparse fill surfaces will get expanded and possibly trimmed.
- std::vector<unsigned char> layer_expansions_and_voids(m_layers.size(), false);
- // start at 1, because the only use is with `layer_expansions_and_voids[layer_idx + 1]`
- for (size_t layer_idx = 1; layer_idx < m_layers.size(); ++layer_idx) {
- const Layer* layer = m_layers[layer_idx];
- bool expansions = false;
- bool voids = false;
- for (const LayerRegion *layerm : layer->regions()) {
- for (const Surface &surface : layerm->fill_surfaces()) {
- if (surface.surface_type == (stPosInternal | stDensSparse))
- voids = true;
- else
- expansions = true;
- if (voids && expansions) {
- layer_expansions_and_voids[layer_idx] = true;
- goto end;
- }
- }
- }
- end:;
- }
- BOOST_LOG_TRIVIAL(debug) << "Collecting surfaces covered with extrusions in parallel - start";
- surfaces_covered.resize(m_layers.size() - 1, Polygons());
- Slic3r::parallel_for(size_t(0), m_layers.size() - 1,
- [this, &surfaces_covered, &layer_expansions_and_voids](const size_t layer_idx) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- if (layer_expansions_and_voids[layer_idx + 1]) {
- // Layer above is partially filled with solid infill (top, bottom, bridging...),
- // while some sparse inill regions are empty (0% infill).
- m_print->throw_if_canceled();
- // Polygons voids;
- // for (const LayerRegion *layerm : m_layers[layer_idx]->regions()) {
- // if (layerm->region().config().fill_density.value == 0.)
- // for (const Surface &surface : layerm->fill_surfaces())
- // // Shrink the holes, let the layer above expand slightly inside the unsupported areas.
- // polygons_append(voids, offset(surface.expolygon, unsupported_width));
- // }
- surfaces_covered[layer_idx] = to_polygons(this->m_layers[layer_idx]->lslices());
- }
- }
- );
- m_print->throw_if_canceled();
- BOOST_LOG_TRIVIAL(debug) << "Collecting surfaces covered with extrusions in parallel - end";
- }
- for (size_t region_id = 0; region_id < this->num_printing_regions(); ++region_id) {
- BOOST_LOG_TRIVIAL(debug) << "Processing external surfaces for region " << region_id << " in parallel - start";
- Slic3r::parallel_for(size_t(0), m_layers.size(),
- [this, &surfaces_covered, region_id](const size_t layer_idx) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- m_print->throw_if_canceled();
- // BOOST_LOG_TRIVIAL(trace) << "Processing external surface, layer" << m_layers[layer_idx]->print_z;
- m_layers[layer_idx]->get_region(int(region_id))->process_external_surfaces(
- // lower layer
- (layer_idx == 0) ? nullptr : m_layers[layer_idx - 1],
- // lower layer polygons with density > 0%
- (layer_idx == 0 || surfaces_covered.empty() || surfaces_covered[layer_idx - 1].empty()) ? nullptr : &surfaces_covered[layer_idx - 1]);
- }
- );
- m_print->throw_if_canceled();
- BOOST_LOG_TRIVIAL(debug) << "Processing external surfaces for region " << region_id << " in parallel - end";
- }
- if (this->has_raft() && ! m_layers.empty()) {
- // Adjust bridge direction of 1st object layer over raft to be perpendicular to the raft contact layer direction.
- Layer &layer = *m_layers.front();
- assert(layer.id() > 0);
- for (LayerRegion *layerm : layer.regions())
- for (Surface &fill : layerm->m_fill_surfaces)
- fill.bridge_angle = -1;
- }
- } // void PrintObject::process_external_surfaces()
- void PrintObject::discover_vertical_shells()
- {
- BOOST_LOG_TRIVIAL(info) << "Discovering vertical shells..." << log_memory_info();
- struct DiscoverVerticalShellsCacheEntry
- {
- // Collected polygons, offsetted
- ExPolygons top_surfaces;
- ExPolygons top_fill_surfaces;
- ExPolygons top_perimeter_surfaces;
- ExPolygons bottom_surfaces;
- ExPolygons bottom_fill_surfaces;
- ExPolygons bottom_perimeter_surfaces;
- ExPolygons holes;
- };
- const bool spiral_vase = this->print()->config().spiral_vase.value;
- const size_t num_layers = spiral_vase ? std::min(size_t(this->printing_region(0).config().bottom_solid_layers), m_layers.size()) : m_layers.size();
- std::vector<DiscoverVerticalShellsCacheEntry> cache_top_botom_regions(num_layers, DiscoverVerticalShellsCacheEntry());
- bool top_bottom_surfaces_all_regions = this->num_printing_regions() > 1 && ! m_config.interface_shells.value;
- // static constexpr const float top_bottom_expansion_coeff = 1.05f;
- // Just a tiny fraction of an infill extrusion width to merge neighbor regions reliably.
- static constexpr const float top_bottom_expansion_coeff = 0.15f; //TODO check if not too little
- if (top_bottom_surfaces_all_regions) {
- // This is a multi-material print and interface_shells are disabled, meaning that the vertical shell thickness
- // is calculated over all materials.
- BOOST_LOG_TRIVIAL(debug) << "Discovering vertical shells in parallel - start : cache top / bottom";
- const size_t num_regions = this->num_printing_regions();
- Slic3r::parallel_for(size_t(0), num_layers,
- [this, &cache_top_botom_regions, num_regions](const size_t idx_layer) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- m_print->throw_if_canceled();
- const Layer& layer = *m_layers[idx_layer];
- DiscoverVerticalShellsCacheEntry& cache = cache_top_botom_regions[idx_layer];
- // Simulate single set of perimeters over all merged regions.
- float perimeter_offset = 0.f;
- float perimeter_min_spacing = FLT_MAX;
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- static size_t debug_idx = 0;
- ++debug_idx;
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- for (size_t region_id = 0; region_id < num_regions; ++ region_id) {
- LayerRegion &layerm = *layer.m_regions[region_id];
- float top_bottom_expansion = float(layerm.flow(frSolidInfill).scaled_spacing()) * top_bottom_expansion_coeff;
- // Top surfaces.
- append(cache.top_surfaces, offset_ex(to_expolygons(layerm.slices().filter_by_type(stPosTop | stDensSolid)), top_bottom_expansion));
- append(cache.top_surfaces, offset_ex(to_expolygons(layerm.fill_surfaces().filter_by_type(stPosTop | stDensSolid)), top_bottom_expansion));
- append(cache.top_fill_surfaces, offset_ex(to_expolygons(layerm.fill_surfaces().filter_by_type(stPosTop | stDensSolid)), top_bottom_expansion));
- append(cache.top_perimeter_surfaces, to_expolygons(layerm.slices().filter_by_type(stPosTop | stDensSolid)));
- // Bottom surfaces.
- auto surfaces_bottom = { stPosBottom | stDensSolid, stPosBottom | stDensSolid | stModBridge };
- append(cache.bottom_surfaces, offset_ex(to_expolygons(layerm.slices().filter_by_types(surfaces_bottom)), top_bottom_expansion));
- append(cache.bottom_surfaces, offset_ex(to_expolygons(layerm.fill_surfaces().filter_by_types(surfaces_bottom)), top_bottom_expansion));
- append(cache.bottom_fill_surfaces, offset_ex(to_expolygons(layerm.fill_surfaces().filter_by_types(surfaces_bottom)), top_bottom_expansion));
- append(cache.bottom_perimeter_surfaces, to_expolygons(layerm.slices().filter_by_type(stPosTop | stDensSolid)));
- // Calculate the maximum perimeter offset as if the slice was extruded with a single extruder only.
- // First find the maxium number of perimeters per region slice.
- unsigned int perimeters = 0;
- for (const Surface &s : layerm.slices())
- perimeters = std::max<unsigned int>(perimeters, s.extra_perimeters);
- perimeters += layerm.region().config().perimeters.value;
- // Then calculate the infill offset.
- if (perimeters > 0) {
- Flow extflow = layerm.flow(frExternalPerimeter);
- Flow flow = layerm.flow(frPerimeter);
- perimeter_offset = std::max(perimeter_offset,
- 0.5f * float(extflow.scaled_width() + extflow.scaled_spacing()) + (float(perimeters) - 1.f) * flow.scaled_spacing());
- perimeter_min_spacing = std::min(perimeter_min_spacing, float(std::min(extflow.scaled_spacing(), flow.scaled_spacing())));
- }
- expolygons_append(cache.holes, layerm.fill_expolygons());
- }
- // Save some computing time by reducing the number of polygons.
- cache.top_surfaces = union_ex(cache.top_surfaces);
- cache.bottom_surfaces = union_ex(cache.bottom_surfaces);
- // For a multi-material print, simulate perimeter / infill split as if only a single extruder has been used for the whole print.
- if (perimeter_offset > 0.) {
- // The layer.lslices are forced to merge by expanding them first.
- expolygons_append(cache.holes, offset2_ex(layer.lslices(), 0.3f * perimeter_min_spacing, -perimeter_offset - 0.3f * perimeter_min_spacing));
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- {
- Slic3r::SVG svg(debug_out_path("discover_vertical_shells-extra-holes-%d.svg", debug_idx), get_extents(layer.lslices()));
- svg.draw(layer.lslices(), "blue");
- svg.draw(union_ex(cache.holes), "red");
- svg.draw_outline(union_ex(cache.holes), "black", "blue", scale_(0.05));
- svg.Close();
- }
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- }
- cache.holes = union_ex(cache.holes);
- }
- );
- m_print->throw_if_canceled();
- BOOST_LOG_TRIVIAL(debug) << "Discovering vertical shells in parallel - end : cache top / bottom";
- }
- for (size_t region_id = 0; region_id < this->num_printing_regions(); ++ region_id) {
-
- const PrintRegion ®ion = this->printing_region(region_id);
- //solid_over_perimeters value, to remove solid fill where there's only perimeters on multiple layers
- const int nb_perimeter_layers_for_solid_fill = region.config().solid_over_perimeters.value;
- const int min_layer_no_solid = region.config().bottom_solid_layers.value - 1;
- const int min_z_no_solid = region.config().bottom_solid_min_thickness;
- if (!top_bottom_surfaces_all_regions) {
- // This is either a single material print, or a multi-material print and interface_shells are enabled, meaning that the vertical shell thickness
- // is calculated over a single material.
- BOOST_LOG_TRIVIAL(debug) << "Discovering vertical shells for region " << region_id << " in parallel - start : cache top / bottom";
- Slic3r::parallel_for(size_t(0), num_layers,
- [this, region_id, &cache_top_botom_regions, nb_perimeter_layers_for_solid_fill, min_layer_no_solid, min_z_no_solid]
- (const size_t idx_layer) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- m_print->throw_if_canceled();
- Layer &layer = *m_layers[idx_layer];
- LayerRegion &layerm = *layer.m_regions[region_id];
- float top_bottom_expansion = float(layerm.flow(frSolidInfill).scaled_spacing()) * top_bottom_expansion_coeff;
- float max_top_bottom_expansion = float(layerm.flow(frSolidInfill).scaled_spacing()) * (1.5f + top_bottom_expansion_coeff);
- // Top surfaces.
- auto& cache = cache_top_botom_regions[idx_layer];
- ExPolygons raw_slice_temp = to_expolygons(layerm.slices().filter_by_type(stPosTop | stDensSolid));
- ExPolygons raw_fill_temp = to_expolygons(layerm.fill_surfaces().filter_by_type(stPosTop | stDensSolid));
- cache.top_surfaces = offset_ex(raw_slice_temp, top_bottom_expansion);
- append(cache.top_surfaces, offset_ex(raw_fill_temp, top_bottom_expansion));
- if (nb_perimeter_layers_for_solid_fill != 0) {
- //it needs to be activated and we don't check the firs layers, where everything have to be solid.
- cache.top_fill_surfaces = offset_ex(raw_fill_temp, max_top_bottom_expansion);
- cache.top_perimeter_surfaces = raw_slice_temp;
- }
- // Bottom surfaces.
- const auto surfaces_bottom = { stPosBottom | stDensSolid, stPosBottom | stDensSolid | stModBridge };
- raw_slice_temp = to_expolygons(layerm.slices().filter_by_types(surfaces_bottom));
- raw_fill_temp = to_expolygons(layerm.fill_surfaces().filter_by_types(surfaces_bottom));
- cache.bottom_surfaces = offset_ex(raw_slice_temp, top_bottom_expansion);
- append(cache.bottom_surfaces, offset_ex(raw_fill_temp, top_bottom_expansion));
- if (nb_perimeter_layers_for_solid_fill != 0) {
- cache.bottom_perimeter_surfaces = raw_slice_temp;
- cache.bottom_fill_surfaces = offset_ex(raw_fill_temp, max_top_bottom_expansion);
- }
- // Holes over all regions. Only collect them once, they are valid for all idx_region iterations.
- if (cache.holes.empty()) {
- for (size_t region_id = 0; region_id < layer.regions().size(); ++ region_id)
- expolygons_append(cache.holes, layer.get_region(region_id)->fill_expolygons());
- }
- }
- );
- m_print->throw_if_canceled();
- BOOST_LOG_TRIVIAL(debug) << "Discovering vertical shells for region " << region_id << " in parallel - end : cache top / bottom";
- }
- BOOST_LOG_TRIVIAL(debug) << "Discovering vertical shells for region " << region_id << " in parallel - start : ensure vertical wall thickness";
-
- Slic3r::parallel_for(size_t(0), num_layers,
- [this, region_id, &cache_top_botom_regions](const size_t idx_layer) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- // printf("discover_vertical_shells for %d \n", idx_layer);
- m_print->throw_if_canceled();
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- static size_t debug_idx = 0;
- ++ debug_idx;
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- Layer *layer = m_layers[idx_layer];
- LayerRegion *layerm = layer->m_regions[region_id];
- const PrintRegionConfig ®ion_config = layerm->region().config();
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- layerm->export_region_slices_to_svg_debug("3_discover_vertical_shells-initial");
- layerm->export_region_fill_surfaces_to_svg_debug("3_discover_vertical_shells-initial");
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- Flow solid_infill_flow = layerm->flow(frSolidInfill);
- coord_t infill_line_spacing = solid_infill_flow.scaled_spacing();
- // Find a union of perimeters below / above this surface to guarantee a minimum shell thickness.
- ExPolygons shell;
- ExPolygons fill_shell; // for nb_perimeter_layers_for_solid_fill
- ExPolygons max_perimeter_shell; // for nb_perimeter_layers_for_solid_fill
- ExPolygons holes;
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- ExPolygons shell_ex;
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- float min_perimeter_infill_spacing = float(infill_line_spacing) * 1.05f;
- const int nb_perimeter_layers_for_solid_fill = region_config.solid_over_perimeters.value;
- const int min_layer_no_solid = region_config.bottom_solid_layers.value - 1;
- const int min_z_no_solid = region_config.bottom_solid_min_thickness;
- #if 0
- // #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- {
- Slic3r::SVG svg_cummulative(debug_out_path("discover_vertical_shells-perimeters-before-union-run%d.svg", debug_idx), this->bounding_box());
- for (int n = (int)idx_layer - n_extra_bottom_layers; n <= (int)idx_layer + n_extra_top_layers; ++ n) {
- if (n < 0 || n >= (int)m_layers.size())
- continue;
- ExPolygons &expolys = m_layers[n]->perimeter_expolygons;
- for (size_t i = 0; i < expolys.size(); ++ i) {
- Slic3r::SVG svg(debug_out_path("discover_vertical_shells-perimeters-before-union-run%d-layer%d-expoly%d.svg", debug_idx, n, i), get_extents(expolys[i]));
- svg.draw(expolys[i]);
- svg.draw_outline(expolys[i].contour, "black", scale_(0.05));
- svg.draw_outline(expolys[i].holes, "blue", scale_(0.05));
- svg.Close();
- svg_cummulative.draw(expolys[i]);
- svg_cummulative.draw_outline(expolys[i].contour, "black", scale_(0.05));
- svg_cummulative.draw_outline(expolys[i].holes, "blue", scale_(0.05));
- }
- }
- }
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- expolygons_append(holes, cache_top_botom_regions[idx_layer].holes);
- auto combine_holes = [&holes](const ExPolygons &holes2) {
- if (holes.empty() || holes2.empty())
- holes.clear();
- else
- holes = intersection_ex(holes, holes2);
- };
- auto combine_shells = [&shell](const ExPolygons &shells2) {
- if (shell.empty())
- shell = std::move(shells2);
- else if (! shells2.empty()) {
- expolygons_append(shell, shells2);
- // Running the union_ using the Clipper library piece by piece is cheaper
- // than running the union_ all at once.
- shell = union_ex(shell);
- }
- };
- static constexpr const bool one_more_layer_below_top_bottom_surfaces = false;
- if (int n_top_layers = region_config.top_solid_layers.value; n_top_layers > 0) {
- // Gather top regions projected to this layer.
- coordf_t print_z = layer->print_z;
- int i = int(idx_layer) + 1;
- int itop = int(idx_layer) + n_top_layers;
- bool at_least_one_top_projected = false;
- for (; i < int(cache_top_botom_regions.size()) &&
- (i < itop || m_layers[i]->print_z - print_z < region_config.top_solid_min_thickness - EPSILON);
- ++ i) {
- at_least_one_top_projected = true;
- const DiscoverVerticalShellsCacheEntry &cache = cache_top_botom_regions[i];
- combine_holes(cache.holes);
- combine_shells(cache.top_surfaces);
- if (nb_perimeter_layers_for_solid_fill != 0 && (idx_layer > min_layer_no_solid || print_z < min_z_no_solid)) {
- if (!cache.top_fill_surfaces.empty()) {
- expolygons_append(fill_shell, cache.top_fill_surfaces);
- fill_shell = union_ex(fill_shell);
- } if (nb_perimeter_layers_for_solid_fill > 1 && i - idx_layer < nb_perimeter_layers_for_solid_fill) {
- expolygons_append(max_perimeter_shell, cache.top_perimeter_surfaces);
- max_perimeter_shell = union_ex(max_perimeter_shell);
- }
- } }
- if (!at_least_one_top_projected && i < int(cache_top_botom_regions.size())) {
- // Lets consider this a special case - with only 1 top solid and minimal shell thickness settings, the
- // boundaries of solid layers are not anchored over/under perimeters, so lets fix it by adding at least one
- // perimeter width of area
- ExPolygons anchor_area = intersection_ex(expand(cache_top_botom_regions[idx_layer].top_surfaces,
- layerm->flow(frExternalPerimeter).scaled_spacing()),
- to_polygons(m_layers[i]->lslices()));
- combine_shells(anchor_area);
- }
- if (one_more_layer_below_top_bottom_surfaces)
- if (i < int(cache_top_botom_regions.size()) &&
- (i <= itop || m_layers[i]->bottom_z() - print_z < region_config.top_solid_min_thickness - EPSILON))
- combine_holes(cache_top_botom_regions[i].holes);
- }
- if (int n_bottom_layers = region_config.bottom_solid_layers.value; n_bottom_layers > 0) {
- // Gather bottom regions projected to this layer.
- coordf_t bottom_z = layer->bottom_z();
- int i = int(idx_layer) - 1;
- int ibottom = int(idx_layer) - n_bottom_layers;
- bool at_least_one_bottom_projected = false;
- for (; i >= 0 &&
- (i > ibottom || bottom_z - m_layers[i]->bottom_z() < region_config.bottom_solid_min_thickness - EPSILON);
- -- i) {
- at_least_one_bottom_projected = true;
- const DiscoverVerticalShellsCacheEntry &cache = cache_top_botom_regions[i];
- combine_holes(cache.holes);
- combine_shells(cache.bottom_surfaces);
- if (nb_perimeter_layers_for_solid_fill != 0 && (idx_layer > min_layer_no_solid || layer->print_z < min_z_no_solid)) {
- if (!cache.bottom_fill_surfaces.empty()) {
- expolygons_append(fill_shell, cache.bottom_fill_surfaces);
- fill_shell = union_ex(fill_shell);
- }
- if (nb_perimeter_layers_for_solid_fill > 1 && idx_layer - i < nb_perimeter_layers_for_solid_fill) {
- expolygons_append(max_perimeter_shell, cache.bottom_perimeter_surfaces);
- max_perimeter_shell = union_ex(max_perimeter_shell);
- }
- }
- }
- if (!at_least_one_bottom_projected && i >= 0) {
- ExPolygons anchor_area = intersection_ex(expand(cache_top_botom_regions[idx_layer].bottom_surfaces,
- layerm->flow(frExternalPerimeter).scaled_spacing()),
- to_polygons(m_layers[i]->lslices()));
- combine_shells(anchor_area);
- }
- if (one_more_layer_below_top_bottom_surfaces)
- if (i >= 0 &&
- (i > ibottom || bottom_z - m_layers[i]->print_z < region_config.bottom_solid_min_thickness - EPSILON))
- combine_holes(cache_top_botom_regions[i].holes);
- }
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- {
- Slic3r::SVG svg(debug_out_path("discover_vertical_shells-perimeters-before-union-%d.svg", debug_idx), get_extents(shell));
- svg.draw(shell);
- svg.draw_outline(shell, "black", scale_(0.05));
- svg.Close();
- }
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- #if 0
- // shell = union_(shell, true);
- shell = union_(shell, false);
- #endif
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- shell_ex = union_safety_offset_ex(shell);
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- //if (shell.empty())
- // continue;
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- {
- Slic3r::SVG svg(debug_out_path("discover_vertical_shells-perimeters-after-union-%d.svg", debug_idx), get_extents(shell));
- svg.draw(shell_ex);
- svg.draw_outline(shell_ex, "black", "blue", scale_(0.05));
- svg.Close();
- }
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- {
- Slic3r::SVG svg(debug_out_path("discover_vertical_shells-internal-wshell-%d.svg", debug_idx), get_extents(shell));
- svg.draw(layerm->fill_surfaces().filter_by_type(stInternal), "yellow", 0.5);
- svg.draw_outline(layerm->fill_surfaces().filter_by_type(stInternal), "black", "blue", scale_(0.05));
- svg.draw(shell_ex, "blue", 0.5);
- svg.draw_outline(shell_ex, "black", "blue", scale_(0.05));
- svg.Close();
- }
- {
- Slic3r::SVG svg(debug_out_path("discover_vertical_shells-internalvoid-wshell-%d.svg", debug_idx), get_extents(shell));
- svg.draw(layerm->fill_surfaces().filter_by_type(stInternalVoid), "yellow", 0.5);
- svg.draw_outline(layerm->fill_surfaces().filter_by_type(stInternalVoid), "black", "blue", scale_(0.05));
- svg.draw(shell_ex, "blue", 0.5);
- svg.draw_outline(shell_ex, "black", "blue", scale_(0.05));
- svg.Close();
- }
- {
- Slic3r::SVG svg(debug_out_path("discover_vertical_shells-internalsolid-wshell-%d.svg", debug_idx), get_extents(shell));
- svg.draw(layerm->fill_surfaces().filter_by_type(stInternalSolid), "yellow", 0.5);
- svg.draw_outline(layerm->fill_surfaces().filter_by_type(stInternalSolid), "black", "blue", scale_(0.05));
- svg.draw(shell_ex, "blue", 0.5);
- svg.draw_outline(shell_ex, "black", "blue", scale_(0.05));
- svg.Close();
- }
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- // Trim the shells region by the internal & internal void surfaces.
- const ExPolygons polygonsInternal = to_expolygons(layerm->fill_surfaces()
- .filter_by_types({ stPosInternal | stDensSparse, stPosInternal | stDensVoid, stPosInternal | stDensSolid }));
- {
- shell = intersection_ex(shell, polygonsInternal, ApplySafetyOffset::Yes);
- expolygons_append(shell, diff_ex(polygonsInternal, holes));
- shell = union_ex(shell);
- //check if a polygon is only over perimeter, in this case evict it (depends from nb_perimeter_layers_for_solid_fill value)
- if (nb_perimeter_layers_for_solid_fill != 0 && (idx_layer > min_layer_no_solid || layer->print_z < min_z_no_solid)) {
- ExPolygons toadd;
- for (int i = 0; i < shell.size(); i++) {
- if (nb_perimeter_layers_for_solid_fill < 2 || intersection_ex(ExPolygons{ shell[i] }, max_perimeter_shell, ApplySafetyOffset::No).empty()) {
- ExPolygons expoly = intersection_ex(ExPolygons{ shell[i] }, fill_shell);
- toadd.insert(toadd.end(), expoly.begin(), expoly.end());
- shell.erase(shell.begin() + i);
- i--;
- }
- }
- expolygons_append(shell, toadd);
- }
- }
- if (shell.empty())
- return; //continue (next layer)
- // Append the internal solids, so they will be merged with the new ones.
- expolygons_append(shell, to_expolygons(layerm->fill_surfaces().filter_by_type(stPosInternal | stDensSolid)));
- // These regions will be filled by a rectilinear full infill. Currently this type of infill
- // only fills regions, which fit at least a single line. To avoid gaps in the sparse infill,
- // make sure that this region does not contain parts narrower than the infill spacing width.
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- Polygons shell_before = shell;
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- ExPolygons regularized_shell;
- {
- // Open to remove (filter out) regions narrower than a bit less than an infill extrusion line width.
- // Such narrow regions are difficult to fill in with a gap fill algorithm (or Arachne), however they are most likely
- // not needed for print stability / quality.
- const float narrow_ensure_vertical_wall_thickness_region_radius = 0.5f * 0.65f * min_perimeter_infill_spacing;
- // Then close gaps narrower than 1.2 * line width, such gaps are difficult to fill in with sparse infill,
- // thus they will be merged into the solid infill.
- const float narrow_sparse_infill_region_radius = 0.5f * 1.2f * min_perimeter_infill_spacing;
- // Finally expand the infill a bit to remove tiny gaps between solid infill and the other regions.
- const float tiny_overlap_radius = 0.2f * min_perimeter_infill_spacing;
- regularized_shell = shrink_ex(offset2_ex(union_ex(shell),
- // Open to remove (filter out) regions narrower than an infill extrusion line width.
- -narrow_ensure_vertical_wall_thickness_region_radius,
- // Then close gaps narrower than 1.2 * line width, such gaps are difficult to fill in with sparse infill.
- narrow_ensure_vertical_wall_thickness_region_radius + narrow_sparse_infill_region_radius, ClipperLib::jtSquare),
- // Finally expand the infill a bit to remove tiny gaps between solid infill and the other regions.
- narrow_sparse_infill_region_radius - tiny_overlap_radius, ClipperLib::jtSquare);
- Polygons object_volume;
- Polygons internal_volume;
- {
- Polygons shrinked_bottom_slice = idx_layer > 0 ? to_polygons(m_layers[idx_layer - 1]->lslices()) : Polygons{};
- Polygons shrinked_upper_slice = (idx_layer + 1) < m_layers.size() ?
- to_polygons(m_layers[idx_layer + 1]->lslices()) :
- Polygons{};
- object_volume = intersection(shrinked_bottom_slice, shrinked_upper_slice);
- //internal_volume = closing(polygonsInternal, float(SCALED_EPSILON));
- internal_volume = offset2(polygonsInternal, float(SCALED_EPSILON), -float(SCALED_EPSILON));
- }
- // The regularization operation may cause scattered tiny drops on the smooth parts of the model, filter them out
- // If the region checks both following conditions, it is removed:
- // 1. the area is very small,
- // OR the area is quite small and it is fully wrapped in model (not visible)
- // the in-model condition is there due to small sloping surfaces, e.g. top of the hull of the benchy
- // 2. the area does not fully cover an internal polygon
- // This is there mainly for a very thin parts, where the solid layers would be missing if the part area is quite small
- regularized_shell.erase(std::remove_if(regularized_shell.begin(), regularized_shell.end(),
- [&internal_volume, &min_perimeter_infill_spacing,
- &object_volume](const ExPolygon &p) {
- return (p.area() < min_perimeter_infill_spacing * scaled(1.5) ||
- (p.area() < min_perimeter_infill_spacing * scaled(8.0) &&
- diff(to_polygons(p), object_volume).empty())) &&
- diff(internal_volume,
- expand(to_polygons(p), min_perimeter_infill_spacing))
- .size() >= internal_volume.size();
- }),
- regularized_shell.end());
- }
- if (regularized_shell.empty())
- return; //continue (next layer)
- ExPolygons new_internal_solid = intersection_ex(polygonsInternal, regularized_shell);
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- {
- Slic3r::SVG svg(debug_out_path("discover_vertical_shells-regularized-%d.svg", debug_idx), get_extents(shell_before));
- // Source shell.
- svg.draw(union_safety_offset_ex(shell_before));
- // Shell trimmed to the internal surfaces.
- svg.draw_outline(union_safety_offset_ex(shell), "black", "blue", scale_(0.05));
- // Regularized infill region.
- svg.draw_outline(new_internal_solid, "red", "magenta", scale_(0.05));
- svg.Close();
- }
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- // Trim the internal & internalvoid by the shell.
- Slic3r::ExPolygons new_internal = diff_ex(to_expolygons(layerm->fill_surfaces().filter_by_type(stPosInternal | stDensSparse)), regularized_shell);
- Slic3r::ExPolygons new_internal_void = diff_ex(to_expolygons(layerm->fill_surfaces().filter_by_type(stPosInternal | stDensVoid)), regularized_shell);
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- {
- SVG::export_expolygons(debug_out_path("discover_vertical_shells-new_internal-%d.svg", debug_idx), get_extents(shell), new_internal, "black", "blue", scale_(0.05));
- SVG::export_expolygons(debug_out_path("discover_vertical_shells-new_internal_void-%d.svg", debug_idx), get_extents(shell), new_internal_void, "black", "blue", scale_(0.05));
- SVG::export_expolygons(debug_out_path("discover_vertical_shells-new_internal_solid-%d.svg", debug_idx), get_extents(shell), new_internal_solid, "black", "blue", scale_(0.05));
- }
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- // Assign resulting internal surfaces to layer.
- layerm->m_fill_surfaces.keep_types({ stPosTop | stDensSolid, stPosBottom | stDensSolid, stPosBottom | stDensSolid | stModBridge });
- //layerm->m_fill_surfaces.keep_types_flag(stPosTop | stPosBottom);
- coord_t scaled_resolution = std::max(SCALED_EPSILON, scale_t(print()->config().resolution.value));
- ensure_valid(new_internal, scaled_resolution);
- ensure_valid(new_internal_void, scaled_resolution);
- ensure_valid(new_internal_solid, scaled_resolution);
- layerm->m_fill_surfaces.append(new_internal, stPosInternal | stDensSparse);
- layerm->m_fill_surfaces.append(new_internal_void, stPosInternal | stDensVoid);
- layerm->m_fill_surfaces.append(new_internal_solid, stPosInternal | stDensSolid);
- } // for each layer
- );
- m_print->throw_if_canceled();
- BOOST_LOG_TRIVIAL(debug) << "Discovering vertical shells for region " << region_id << " in parallel - end";
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- for (size_t idx_layer = 0; idx_layer < m_layers.size(); ++idx_layer) {
- LayerRegion *layerm = m_layers[idx_layer]->get_region(region_id);
- layerm->export_region_slices_to_svg_debug("3_discover_vertical_shells-final");
- layerm->export_region_fill_surfaces_to_svg_debug("3_discover_vertical_shells-final");
- }
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- } // for each region
- } // void PrintObject::discover_vertical_shells()
- // #define DEBUG_BRIDGE_OVER_INFILL
- #ifdef DEBUG_BRIDGE_OVER_INFILL
- template<typename T> void debug_draw(std::string name, const T& a, const T& b, const T& c, const T& d)
- {
- std::vector<std::string> colors = {"red", "green", "blue", "orange"};
- BoundingBox bbox = get_extents(a);
- bbox.merge(get_extents(b));
- bbox.merge(get_extents(c));
- bbox.merge(get_extents(d));
- bbox.offset(scale_(1.));
- ::Slic3r::SVG svg(debug_out_path(name.c_str()).c_str(), bbox);
- svg.draw(a, colors[0], scale_(0.3));
- svg.draw(b, colors[1], scale_(0.23));
- svg.draw(c, colors[2], scale_(0.16));
- svg.draw(d, colors[3], scale_(0.10));
- svg.Close();
- }
- #endif
- /* This method applies overextrude flow to the first internal solid layer above
- bridge (which is over sparse infill) note: it's almost complete copy/paste from the method behind,
- i think it should be merged before gitpull that.
- */
- void PrintObject::replaceSurfaceType(SurfaceType st_to_replace, SurfaceType st_replacement, SurfaceType st_under_it)
- {
- BOOST_LOG_TRIVIAL(info) << "overextrude over Bridge...";
- coord_t scaled_resolution = std::max(SCALED_EPSILON, scale_t(print()->config().resolution.value));
- for (size_t region_id = 0; region_id < this->num_printing_regions(); ++region_id) {
- const PrintRegion& region = this->printing_region(region_id);
- // skip over-bridging in case there are no modification
- if (region.config().over_bridge_flow_ratio.get_abs_value(1) == 1) continue;
- for (LayerPtrs::iterator layer_it = m_layers.begin(); layer_it != m_layers.end(); ++layer_it) {
- // skip first layer
- if (layer_it == this->layers().begin()) continue;
- Layer* layer = *layer_it;
- LayerRegion* layerm = layer->regions()[region_id];
- Polygons poly_to_check;
- // extract the surfaces that might be transformed
- layerm->fill_surfaces().filter_by_type(st_to_replace, &poly_to_check);
- Polygons poly_to_replace = poly_to_check;
- // check the lower layer
- if (int(layer_it - this->layers().begin()) - 1 >= 0) {
- const Layer* lower_layer = this->layers()[int(layer_it - this->layers().begin()) - 1];
- // iterate through regions and collect internal surfaces
- Polygons lower_internal;
- for (LayerRegion* lower_layerm : lower_layer->m_regions) {
- lower_layerm->fill_surfaces().filter_by_type(st_under_it, &lower_internal);
- }
- // intersect such lower internal surfaces with the candidate solid surfaces
- poly_to_replace = intersection(poly_to_replace, lower_internal);
- }
- if (poly_to_replace.empty()) continue;
- // compute the remaning internal solid surfaces as difference
- ExPolygons not_expoly_to_replace = ensure_valid(diff_ex(poly_to_check, poly_to_replace, ApplySafetyOffset::Yes), scaled_resolution);
- // build the new collection of fill_surfaces
- {
- Surfaces new_surfaces;
- for (Surfaces::const_iterator surface = layerm->fill_surfaces().surfaces.begin(); surface != layerm->fill_surfaces().surfaces.end(); ++surface) {
- if (surface->surface_type != st_to_replace) {
- surface->expolygon.assert_valid();
- new_surfaces.push_back(*surface);
- }
- }
- for (ExPolygon& ex : ensure_valid(union_ex(poly_to_replace), scaled_resolution)) {
- ex.assert_valid();
- new_surfaces.push_back(Surface(st_replacement, ex));
- }
- for (ExPolygon& ex : not_expoly_to_replace) {
- ex.assert_valid();
- new_surfaces.push_back(Surface(st_to_replace, ex));
- }
- layerm->set_fill_surfaces().surfaces = new_surfaces;
- }
- }
- }
- }
- // This method applies bridge flow to the first internal solid layer above sparse infill.
- void PrintObject::bridge_over_infill()
- {
- BOOST_LOG_TRIVIAL(info) << "Bridge over infill - Start" << log_memory_info();
- struct CandidateSurface
- {
- CandidateSurface(const Surface *original_surface,
- int layer_index,
- Polygons new_polys,
- const LayerRegion *region,
- double bridge_angle)
- : original_surface(original_surface)
- , layer_index(layer_index)
- , new_polys(new_polys)
- , region(region)
- , bridge_angle(bridge_angle)
- {}
- const Surface *original_surface;
- int layer_index;
- Polygons new_polys;
- const LayerRegion *region;
- double bridge_angle;
- };
- std::map<size_t, std::vector<CandidateSurface>> surfaces_by_layer;
- // SECTION to gather and filter surfaces for expanding, and then cluster them by layer
- {
- tbb::concurrent_vector<CandidateSurface> candidate_surfaces;
- Slic3r::parallel_for(size_t(0), this->layers().size(),
- [po = static_cast<const PrintObject *>(this), &candidate_surfaces](const size_t lidx) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- const Layer *layer = po->get_layer(lidx);
- if (layer->lower_layer != nullptr) {
- double spacing = layer->regions().front()->flow(frSolidInfill).scaled_spacing();
- // unsupported area will serve as a filter for polygons worth bridging.
- Polygons unsupported_area;
- Polygons lower_layer_solids;
- for (const LayerRegion *region : layer->lower_layer->regions()) {
- Polygons fill_polys = to_polygons(region->fill_expolygons());
- // initially consider the whole layer unsupported, but also gather solid layers to later cut off supported parts
- unsupported_area.insert(unsupported_area.end(), fill_polys.begin(), fill_polys.end());
- for (const Surface &surface : region->fill_surfaces()) {
- // > 80 instead of ==100, because at 80% it's like solid for bridge, it doesn't have space.
- if ( (surface.surface_type & stDensSolid) == stDensSolid || region->region().config().fill_density.value > 80) {
- Polygons p = to_polygons(surface.expolygon);
- lower_layer_solids.insert(lower_layer_solids.end(), p.begin(), p.end());
- }
- }
- }
- unsupported_area = closing(unsupported_area, float(SCALED_EPSILON));
- // By expanding the lower layer solids, we avoid making bridges from the tiny internal overhangs that are (very likely) supported by previous layer solids
- // NOTE that we cannot filter out polygons worth bridging by their area, because sometimes there is a very small internal island that will grow into large hole
- lower_layer_solids = shrink(lower_layer_solids, 1 * spacing); // first remove thin regions that will not support anything
- lower_layer_solids = expand(lower_layer_solids, (1 + 3) * spacing); // then expand back (opening), and further for parts supported by internal solids
- // By shrinking the unsupported area, we avoid making bridges from narrow ensuring region along perimeters.
- unsupported_area = shrink(unsupported_area, 3 * spacing);
- unsupported_area = diff(unsupported_area, lower_layer_solids);
- for (const LayerRegion *region : layer->regions()) {
- SurfacesPtr region_internal_solids = region->fill_surfaces().filter_by_type(stPosInternal | stDensSolid);
- for (const Surface *s : region_internal_solids) {
- Polygons unsupported = intersection(to_polygons(s->expolygon), unsupported_area);
- // The following flag marks those surfaces, which overlap with unuspported area, but at least part of them is supported.
- // These regions can be filtered by area, because they for sure are touching solids on lower layers, and it does not make sense to bridge their tiny overhangs
- bool partially_supported = area(unsupported) < area(to_polygons(s->expolygon)) - EPSILON;
- if (!unsupported.empty() && (!partially_supported || area(unsupported) > 3 * 3 * spacing * spacing)) {
- Polygons worth_bridging = intersection(to_polygons(s->expolygon), expand(unsupported, 4 * spacing));
- // after we extracted the part worth briding, we go over the leftovers and merge the tiny ones back, to not brake the surface too much
- for (const Polygon& p : diff(to_polygons(s->expolygon), expand(worth_bridging, spacing))) {
- double area = p.area();
- if (area < spacing * scale_(12.0) && area > spacing * spacing) {
- worth_bridging.push_back(p);
- }
- }
- worth_bridging = intersection(closing(worth_bridging, float(SCALED_EPSILON)), s->expolygon);
- candidate_surfaces.push_back(CandidateSurface(s, lidx, worth_bridging, region, 0));
- #ifdef DEBUG_BRIDGE_OVER_INFILL
- debug_draw(std::to_string(lidx) + "_candidate_surface_" + std::to_string(area(s->expolygon)),
- to_lines(region->layer()->lslices()), to_lines(s->expolygon), to_lines(worth_bridging),
- to_lines(unsupported_area));
- #endif
- #ifdef DEBUG_BRIDGE_OVER_INFILL
- debug_draw(std::to_string(lidx) + "_candidate_processing_" + std::to_string(area(unsupported)),
- to_lines(unsupported), to_lines(intersection(to_polygons(s->expolygon), expand(unsupported, 5 * spacing))),
- to_lines(diff(to_polygons(s->expolygon), expand(worth_bridging, spacing))),
- to_lines(unsupported_area));
- #endif
- }
- }
- }
- }
- });
- for (const CandidateSurface &c : candidate_surfaces) {
- surfaces_by_layer[c.layer_index].push_back(c);
- }
- }
- // LIGHTNING INFILL SECTION - If lightning infill is used somewhere, we check the areas that are going to be bridges, and those that rely on the
- // lightning infill under them get expanded. This somewhat helps to ensure that most of the extrusions are anchored to the lightning infill at the ends.
- // It requires modifying this instance of print object in a specific way, so that we do not invalidate the pointers in our surfaces_by_layer structure.
- bool has_lightning_infill = false;
- for (size_t i = 0; i < this->num_printing_regions(); i++) {
- if (this->printing_region(i).config().fill_pattern.value == InfillPattern::ipLightning) {
- has_lightning_infill = true;
- break;
- }
- }
- if (has_lightning_infill) {
- // Prepare backup data for the Layer Region infills. Before modfiyng the layer region, we backup its fill surfaces by moving! them into this map.
- // then a copy is created, modifiyed and passed to lightning infill generator. After generator is created, we restore the original state of the fills
- // again by moving the data from this map back to the layer regions. This ensures that pointers to surfaces stay valid.
- std::map<size_t, std::map<const LayerRegion *, SurfaceCollection>> backup_surfaces;
- for (size_t lidx = 0; lidx < this->layer_count(); lidx++) {
- backup_surfaces[lidx] = {};
- }
-
- Slic3r::parallel_for(size_t(0), this->layers().size(),
- [po = this, &backup_surfaces, &surfaces_by_layer](const size_t lidx) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- if (surfaces_by_layer.find(lidx) == surfaces_by_layer.end())
- return; //continue (next layer)
- Layer *layer = po->get_layer(lidx);
- const Layer *lower_layer = layer->lower_layer;
- if (lower_layer == nullptr)
- return; //continue (next layer)
- Polygons lightning_fill;
- for (const LayerRegion *region : lower_layer->regions()) {
- if (region->region().config().fill_pattern.value == InfillPattern::ipLightning) {
- Polygons lf = to_polygons(region->fill_surfaces().filter_by_type(stPosInternal | stDensSparse));
- lightning_fill.insert(lightning_fill.end(), lf.begin(), lf.end());
- }
- }
- if (lightning_fill.empty())
- return; //continue (next layer)
- for (LayerRegion *region : layer->regions()) {
- backup_surfaces[lidx][region] = std::move(
- region->m_fill_surfaces); // Make backup copy by move!! so that pointers in candidate surfaces stay valid
- // Copy the surfaces back, this will make copy, but we will later discard it anyway
- region->m_fill_surfaces = backup_surfaces[lidx][region];
- for(auto &srf : region->m_fill_surfaces)
- srf.expolygon.assert_valid();
- }
- for (LayerRegion *region : layer->regions()) {
- ExPolygons sparse_infill = to_expolygons(region->fill_surfaces().filter_by_type(stPosInternal | stDensSparse));
- ExPolygons solid_infill = to_expolygons(region->fill_surfaces().filter_by_type(stPosInternal | stDensSolid));
- if (sparse_infill.empty()) {
- break;
- }
- for (const auto &surface : surfaces_by_layer[lidx]) {
- if (surface.region != region)
- continue;
- ExPolygons expansion = intersection_ex(sparse_infill, expand(surface.new_polys, scaled<float>(3.0)));
- solid_infill.insert(solid_infill.end(), expansion.begin(), expansion.end());
- }
- solid_infill = union_safety_offset_ex(solid_infill);
- sparse_infill = diff_ex(sparse_infill, solid_infill);
- region->m_fill_surfaces.remove_types({stPosInternal | stDensSolid, stPosInternal | stDensSparse});
- for (const ExPolygon &ep : solid_infill) {
- ep.assert_valid();
- region->m_fill_surfaces.surfaces.emplace_back(stPosInternal | stDensSolid, ep);
- }
- for (const ExPolygon &ep : sparse_infill) {
- ep.assert_valid();
- region->m_fill_surfaces.surfaces.emplace_back(stPosInternal | stDensSparse, ep);
- }
- }
- }
- );
- // Use the modified surfaces to generate expanded lightning anchors
- this->m_lightning_generator = this->prepare_lightning_infill_data();
- // And now restore carefully the original surfaces, again using move to avoid reallocation and preserving the validity of the
- // pointers in surface candidates
- for (size_t lidx = 0; lidx < this->layer_count(); lidx++) {
- Layer *layer = this->get_layer(lidx);
- for (LayerRegion *region : layer->regions()) {
- if (backup_surfaces[lidx].find(region) != backup_surfaces[lidx].end()) {
- region->m_fill_surfaces = std::move(backup_surfaces[lidx][region]);
- for(auto &srf : region->m_fill_surfaces)
- srf.expolygon.assert_valid();
- }
- }
- }
- }
- std::map<size_t, Polylines> infill_lines;
- // SECTION to generate infill polylines
- {
- std::vector<std::pair<const Surface *, float>> surfaces_w_bottom_z;
- for (const auto &pair : surfaces_by_layer) {
- for (const CandidateSurface &c : pair.second) {
- surfaces_w_bottom_z.emplace_back(c.original_surface, c.region->m_layer->bottom_z());
- }
- }
- this->m_adaptive_fill_octrees = this->prepare_adaptive_infill_data(surfaces_w_bottom_z);
- std::vector<size_t> layers_to_generate_infill;
- for (const auto &pair : surfaces_by_layer) {
- assert(pair.first > 0);
- infill_lines[pair.first - 1] = {};
- layers_to_generate_infill.push_back(pair.first - 1);
- }
-
- Slic3r::parallel_for(size_t(0), layers_to_generate_infill.size(),
- [po = static_cast<const PrintObject *>(this), &layers_to_generate_infill, &infill_lines]
- (const size_t job_idx) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- size_t lidx = layers_to_generate_infill[job_idx];
- infill_lines.at(
- lidx) = po->get_layer(lidx)->generate_sparse_infill_polylines_for_anchoring(po->m_adaptive_fill_octrees.first.get(),
- po->m_adaptive_fill_octrees.second.get(),
- po->m_lightning_generator.get());
- }
- );
- #ifdef DEBUG_BRIDGE_OVER_INFILL
- for (const auto &il : infill_lines) {
- debug_draw(std::to_string(il.first) + "_infill_lines", to_lines(get_layer(il.first)->lslices()), to_lines(il.second), {}, {});
- }
- #endif
- }
- // cluster layers by depth needed for thick bridges. Each cluster is to be processed by single thread sequentially, so that bridges cannot appear one on another
- std::vector<std::vector<size_t>> clustered_layers_for_threads;
- float target_flow_height_factor = 0.9f;
- {
- std::vector<size_t> layers_with_candidates;
- std::map<size_t, Polygons> layer_area_covered_by_candidates;
- for (const auto& pair : surfaces_by_layer) {
- layers_with_candidates.push_back(pair.first);
- layer_area_covered_by_candidates[pair.first] = {};
- }
- // prepare inflated filter for each candidate on each layer. layers will be put into single thread cluster if they are close to each other (z-axis-wise)
- // and if the inflated AABB polygons overlap somewhere
- Slic3r::parallel_for(size_t(0), layers_with_candidates.size(),
- [&layers_with_candidates, &surfaces_by_layer, &layer_area_covered_by_candidates]
- (const size_t job_idx) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- size_t lidx = layers_with_candidates[job_idx];
- for (const auto &candidate : surfaces_by_layer.at(lidx)) {
- Polygon candiate_inflated_aabb = get_extents(candidate.new_polys).inflated(scale_(7)).polygon();
- layer_area_covered_by_candidates.at(lidx) = union_(layer_area_covered_by_candidates.at(lidx),
- Polygons{candiate_inflated_aabb});
- }
- }
- );
- // note: surfaces_by_layer is ordered map
- for (auto pair : surfaces_by_layer) {
- const LayerRegion *first_lregion = this->get_layer(pair.first)->regions()[0];
- if (clustered_layers_for_threads.empty() ||
- this->get_layer(clustered_layers_for_threads.back().back())->print_z <
- this->get_layer(pair.first)->print_z -
- first_lregion->bridging_flow(frSolidInfill, first_lregion->region().config().bridge_type.value).height() * target_flow_height_factor -
- EPSILON ||
- intersection(layer_area_covered_by_candidates[clustered_layers_for_threads.back().back()],
- layer_area_covered_by_candidates[pair.first])
- .empty()) {
- clustered_layers_for_threads.push_back({pair.first});
- } else {
- clustered_layers_for_threads.back().push_back(pair.first);
- }
- }
- #ifdef DEBUG_BRIDGE_OVER_INFILL
- std::cout << "BRIDGE OVER INFILL CLUSTERED LAYERS FOR SINGLE THREAD" << std::endl;
- for (auto cluster : clustered_layers_for_threads) {
- std::cout << "CLUSTER: ";
- for (auto l : cluster) {
- std::cout << l << " ";
- }
- std::cout << std::endl;
- }
- #endif
- }
- // LAMBDA to gather areas with sparse infill deep enough that we can fit thick bridges there.
- auto gather_areas_w_depth = [target_flow_height_factor](const PrintObject *po, int lidx, float target_flow_height) {
- // Gather layers sparse infill areas, to depth defined by used bridge flow
- ExPolygons layers_sparse_infill{};
- ExPolygons not_sparse_infill{};
- double bottom_z = po->get_layer(lidx)->print_z - target_flow_height * target_flow_height_factor - EPSILON;
- for (int i = int(lidx) - 1; i >= 0; --i) {
- // Stop iterating if layer is lower than bottom_z and at least one iteration was made
- const Layer *layer = po->get_layer(i);
- if (layer->print_z < bottom_z && i < int(lidx) - 1)
- break;
- for (const LayerRegion *region : layer->regions()) {
- bool has_low_density = region->region().config().fill_density.value < 100;
- for (const Surface &surface : region->fill_surfaces()) {
- if ((surface.surface_type == (stPosInternal | stDensSparse) && has_low_density) || surface.surface_type == (stPosInternal | stDensVoid) ) {
- layers_sparse_infill.push_back(surface.expolygon);
- } else {
- not_sparse_infill.push_back(surface.expolygon);
- }
- }
- }
- }
- layers_sparse_infill = union_ex(layers_sparse_infill);
- layers_sparse_infill = closing_ex(layers_sparse_infill, float(SCALED_EPSILON));
- not_sparse_infill = union_ex(not_sparse_infill);
- not_sparse_infill = closing_ex(not_sparse_infill, float(SCALED_EPSILON));
- return diff(layers_sparse_infill, not_sparse_infill);
- };
- // LAMBDA do determine optimal bridging angle
- auto determine_bridging_angle = [](const Polygons &bridged_area, const Lines &anchors, InfillPattern dominant_pattern) {
- AABBTreeLines::LinesDistancer<Line> lines_tree(anchors);
- std::map<double, int> counted_directions;
- for (const Polygon &p : bridged_area) {
- double acc_distance = 0;
- for (int point_idx = 0; point_idx < int(p.points.size()) - 1; ++point_idx) {
- Vec2d start = p.points[point_idx].cast<double>();
- Vec2d next = p.points[point_idx + 1].cast<double>();
- Vec2d v = next - start; // vector from next to current
- double dist_to_next = v.norm();
- acc_distance += dist_to_next;
- if (acc_distance > scaled(2.0)) {
- acc_distance = 0.0;
- v.normalize();
- int lines_count = int(std::ceil(dist_to_next / scaled(2.0)));
- float step_size = dist_to_next / lines_count;
- for (int i = 0; i < lines_count; ++i) {
- Point a = (start + v * (i * step_size)).cast<coord_t>();
- auto [distance, index, p] = lines_tree.distance_from_lines_extra<false>(a);
- double angle = lines_tree.get_line(index).orientation();
- if (angle > PI) {
- angle -= PI;
- }
- angle += PI * 0.5;
- counted_directions[angle]++;
- }
- }
- }
- }
- std::pair<double, int> best_dir{0, 0};
- // sliding window accumulation
- for (const auto &dir : counted_directions) {
- int score_acc = 0;
- double dir_acc = 0;
- double window_start_angle = dir.first - PI * 0.1;
- double window_end_angle = dir.first + PI * 0.1;
- for (auto dirs_window = counted_directions.lower_bound(window_start_angle);
- dirs_window != counted_directions.upper_bound(window_end_angle); dirs_window++) {
- dir_acc += dirs_window->first * dirs_window->second;
- score_acc += dirs_window->second;
- }
- // current span of directions is 0.5 PI to 1.5 PI (due to the aproach.). Edge values should also account for the
- // opposite direction.
- if (window_start_angle < 0.5 * PI) {
- for (auto dirs_window = counted_directions.lower_bound(1.5 * PI - (0.5 * PI - window_start_angle));
- dirs_window != counted_directions.end(); dirs_window++) {
- dir_acc += dirs_window->first * dirs_window->second;
- score_acc += dirs_window->second;
- }
- }
- if (window_start_angle > 1.5 * PI) {
- for (auto dirs_window = counted_directions.begin();
- dirs_window != counted_directions.upper_bound(window_start_angle - 1.5 * PI); dirs_window++) {
- dir_acc += dirs_window->first * dirs_window->second;
- score_acc += dirs_window->second;
- }
- }
- if (score_acc > best_dir.second) {
- best_dir = {dir_acc / score_acc, score_acc};
- }
- }
- double bridging_angle = best_dir.first;
- if (bridging_angle == 0) {
- bridging_angle = 0.001;
- }
- switch (dominant_pattern) {
- case ipHilbertCurve: bridging_angle += 0.25 * PI; break;
- case ipOctagramSpiral: bridging_angle += (1.0 / 16.0) * PI; break;
- default: break;
- }
- return bridging_angle;
- };
- // LAMBDA that will fill given polygons with lines, exapand the lines to the nearest anchor, and reconstruct polygons from the newly
- // generated lines
- auto construct_anchored_polygon = [](Polygons bridged_area, Lines anchors, const Flow &bridging_flow, double bridging_angle) {
- auto lines_rotate = [](Lines &lines, double cos_angle, double sin_angle) {
- for (Line &l : lines) {
- double ax = double(l.a.x());
- double ay = double(l.a.y());
- l.a.x() = coord_t(round(cos_angle * ax - sin_angle * ay));
- l.a.y() = coord_t(round(cos_angle * ay + sin_angle * ax));
- double bx = double(l.b.x());
- double by = double(l.b.y());
- l.b.x() = coord_t(round(cos_angle * bx - sin_angle * by));
- l.b.y() = coord_t(round(cos_angle * by + sin_angle * bx));
- }
- };
- auto segments_overlap = [](coord_t alow, coord_t ahigh, coord_t blow, coord_t bhigh) {
- return (alow >= blow && alow <= bhigh) || (ahigh >= blow && ahigh <= bhigh) || (blow >= alow && blow <= ahigh) ||
- (bhigh >= alow && bhigh <= ahigh);
- };
- Polygons expanded_bridged_area{};
- double aligning_angle = -bridging_angle + PI * 0.5;
- {
- polygons_rotate(bridged_area, aligning_angle);
- lines_rotate(anchors, cos(aligning_angle), sin(aligning_angle));
- BoundingBox bb_x = get_extents(bridged_area);
- BoundingBox bb_y = get_extents(anchors);
- const size_t n_vlines = (bb_x.max.x() - bb_x.min.x() + bridging_flow.scaled_spacing() - 1) / bridging_flow.scaled_spacing();
- std::vector<Line> vertical_lines(n_vlines);
- for (size_t i = 0; i < n_vlines; i++) {
- coord_t x = bb_x.min.x() + i * bridging_flow.scaled_spacing();
- coord_t y_min = bb_y.min.y() - bridging_flow.scaled_spacing();
- coord_t y_max = bb_y.max.y() + bridging_flow.scaled_spacing();
- vertical_lines[i].a = Point{x, y_min};
- vertical_lines[i].b = Point{x, y_max};
- }
- auto anchors_and_walls_tree = AABBTreeLines::LinesDistancer<Line>{std::move(anchors)};
- auto bridged_area_tree = AABBTreeLines::LinesDistancer<Line>{to_lines(bridged_area)};
- std::vector<std::vector<Line>> polygon_sections(n_vlines);
- for (size_t i = 0; i < n_vlines; i++) {
- auto area_intersections = bridged_area_tree.intersections_with_line<true>(vertical_lines[i]);
- for (int intersection_idx = 0; intersection_idx < int(area_intersections.size()) - 1; intersection_idx++) {
- if (bridged_area_tree.outside(
- (area_intersections[intersection_idx].first + area_intersections[intersection_idx + 1].first) / 2) < 0) {
- polygon_sections[i].emplace_back(area_intersections[intersection_idx].first,
- area_intersections[intersection_idx + 1].first);
- }
- }
- auto anchors_intersections = anchors_and_walls_tree.intersections_with_line<true>(vertical_lines[i]);
- for (Line §ion : polygon_sections[i]) {
- auto maybe_below_anchor = std::upper_bound(anchors_intersections.rbegin(), anchors_intersections.rend(), section.a,
- [](const Point &a, const std::pair<Point, size_t> &b) {
- return a.y() > b.first.y();
- });
- if (maybe_below_anchor != anchors_intersections.rend()) {
- section.a = maybe_below_anchor->first;
- section.a.y() -= bridging_flow.scaled_width() * (0.5 + 0.5);
- }
- auto maybe_upper_anchor = std::upper_bound(anchors_intersections.begin(), anchors_intersections.end(), section.b,
- [](const Point &a, const std::pair<Point, size_t> &b) {
- return a.y() < b.first.y();
- });
- if (maybe_upper_anchor != anchors_intersections.end()) {
- section.b = maybe_upper_anchor->first;
- section.b.y() += bridging_flow.scaled_width() * (0.5 + 0.5);
- }
- }
- for (int section_idx = 0; section_idx < int(polygon_sections[i].size()) - 1; section_idx++) {
- Line §ion_a = polygon_sections[i][section_idx];
- Line §ion_b = polygon_sections[i][section_idx + 1];
- if (segments_overlap(section_a.a.y(), section_a.b.y(), section_b.a.y(), section_b.b.y())) {
- section_b.a = section_a.a.y() < section_b.a.y() ? section_a.a : section_b.a;
- section_b.b = section_a.b.y() < section_b.b.y() ? section_b.b : section_a.b;
- section_a.a = section_a.b;
- }
- }
- polygon_sections[i].erase(std::remove_if(polygon_sections[i].begin(), polygon_sections[i].end(),
- [](const Line &s) { return s.a == s.b; }),
- polygon_sections[i].end());
- std::sort(polygon_sections[i].begin(), polygon_sections[i].end(),
- [](const Line &a, const Line &b) { return a.a.y() < b.b.y(); });
- }
- // reconstruct polygon from polygon sections
- struct TracedPoly
- {
- Points lows;
- Points highs;
- };
- std::vector<TracedPoly> current_traced_polys;
- for (const auto &polygon_slice : polygon_sections) {
- std::unordered_set<const Line *> used_segments;
- for (TracedPoly &traced_poly : current_traced_polys) {
- auto candidates_begin = std::upper_bound(polygon_slice.begin(), polygon_slice.end(), traced_poly.lows.back(),
- [](const Point &low, const Line &seg) { return seg.b.y() > low.y(); });
- auto candidates_end = std::upper_bound(polygon_slice.begin(), polygon_slice.end(), traced_poly.highs.back(),
- [](const Point &high, const Line &seg) { return seg.a.y() > high.y(); });
- bool segment_added = false;
- for (auto candidate = candidates_begin; candidate != candidates_end && !segment_added; candidate++) {
- if (used_segments.find(&(*candidate)) != used_segments.end()) {
- continue;
- }
- if ((traced_poly.lows.back() - candidate->a).cast<double>().squaredNorm() <
- 36.0 * double(bridging_flow.scaled_spacing()) * bridging_flow.scaled_spacing()) {
- traced_poly.lows.push_back(candidate->a);
- } else {
- traced_poly.lows.push_back(traced_poly.lows.back() + Point{bridging_flow.scaled_spacing() / 2, 0});
- traced_poly.lows.push_back(candidate->a - Point{bridging_flow.scaled_spacing() / 2, 0});
- traced_poly.lows.push_back(candidate->a);
- }
- if ((traced_poly.highs.back() - candidate->b).cast<double>().squaredNorm() <
- 36.0 * double(bridging_flow.scaled_spacing()) * bridging_flow.scaled_spacing()) {
- traced_poly.highs.push_back(candidate->b);
- } else {
- traced_poly.highs.push_back(traced_poly.highs.back() + Point{bridging_flow.scaled_spacing() / 2, 0});
- traced_poly.highs.push_back(candidate->b - Point{bridging_flow.scaled_spacing() / 2, 0});
- traced_poly.highs.push_back(candidate->b);
- }
- segment_added = true;
- used_segments.insert(&(*candidate));
- }
- if (!segment_added) {
- // Zero overlapping segments, we just close this polygon
- traced_poly.lows.push_back(traced_poly.lows.back() + Point{bridging_flow.scaled_spacing() / 2, 0});
- traced_poly.highs.push_back(traced_poly.highs.back() + Point{bridging_flow.scaled_spacing() / 2, 0});
- Polygon &new_poly = expanded_bridged_area.emplace_back(std::move(traced_poly.lows));
- new_poly.points.insert(new_poly.points.end(), traced_poly.highs.rbegin(), traced_poly.highs.rend());
- traced_poly.lows.clear();
- traced_poly.highs.clear();
- }
- }
- current_traced_polys.erase(std::remove_if(current_traced_polys.begin(), current_traced_polys.end(),
- [](const TracedPoly &tp) { return tp.lows.empty(); }),
- current_traced_polys.end());
- for (const auto &segment : polygon_slice) {
- if (used_segments.find(&segment) == used_segments.end()) {
- TracedPoly &new_tp = current_traced_polys.emplace_back();
- new_tp.lows.push_back(segment.a - Point{bridging_flow.scaled_spacing() / 2, 0});
- new_tp.lows.push_back(segment.a);
- new_tp.highs.push_back(segment.b - Point{bridging_flow.scaled_spacing() / 2, 0});
- new_tp.highs.push_back(segment.b);
- }
- }
- }
- // add not closed polys
- for (TracedPoly &traced_poly : current_traced_polys) {
- Polygon &new_poly = expanded_bridged_area.emplace_back(std::move(traced_poly.lows));
- new_poly.points.insert(new_poly.points.end(), traced_poly.highs.rbegin(), traced_poly.highs.rend());
- }
- expanded_bridged_area = union_safety_offset(expanded_bridged_area);
- }
- polygons_rotate(expanded_bridged_area, -aligning_angle);
- return expanded_bridged_area;
- };
-
- Slic3r::parallel_for(size_t(0), clustered_layers_for_threads.size(),
- [po = static_cast<const PrintObject *>(this), target_flow_height_factor, &surfaces_by_layer, &clustered_layers_for_threads,
- gather_areas_w_depth, &infill_lines, determine_bridging_angle, construct_anchored_polygon]
- (const size_t cluster_idx) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- for (size_t job_idx = 0; job_idx < clustered_layers_for_threads[cluster_idx].size(); job_idx++) {
- size_t lidx = clustered_layers_for_threads[cluster_idx][job_idx];
- const Layer *layer = po->get_layer(lidx);
- // this thread has exclusive access to all surfaces in layers enumerated in
- // clustered_layers_for_threads[cluster_idx]
- // Presort the candidate polygons. This will help choose the same angle for neighbournig surfaces, that
- // would otherwise compete over anchoring sparse infill lines, leaving one area unachored
- std::sort(surfaces_by_layer[lidx].begin(), surfaces_by_layer[lidx].end(),
- [](const CandidateSurface &left, const CandidateSurface &right) {
- auto a = get_extents(left.new_polys);
- auto b = get_extents(right.new_polys);
- if (a.min.x() == b.min.x()) {
- return a.min.y() < b.min.y();
- };
- return a.min.x() < b.min.x();
- });
- if (surfaces_by_layer[lidx].size() > 2) {
- Vec2d origin = get_extents(surfaces_by_layer[lidx].front().new_polys).max.cast<double>();
- std::stable_sort(surfaces_by_layer[lidx].begin() + 1, surfaces_by_layer[lidx].end(),
- [origin](const CandidateSurface &left, const CandidateSurface &right) {
- auto a = get_extents(left.new_polys);
- auto b = get_extents(right.new_polys);
- return (origin - a.min.cast<double>()).squaredNorm() <
- (origin - b.min.cast<double>()).squaredNorm();
- });
- }
- // Gather deep infill areas, where thick bridges fit
- const LayerRegion *first_lregion = surfaces_by_layer[lidx].front().region;
- Flow bridge_flow = first_lregion->bridging_flow(frSolidInfill, first_lregion->region().config().bridge_type);
- const coord_t spacing = bridge_flow.scaled_spacing();
- const float bridge_height = std::max(float(layer->height), bridge_flow.height());
- //const coord_t bridge_width = bridge_flow.scaled_width();
- const coord_t target_flow_height = bridge_flow.height() * target_flow_height_factor;
- Polygons deep_infill_area = gather_areas_w_depth(po, lidx, target_flow_height);
- {
- // Now also remove area that has been already filled on lower layers by bridging expansion - For this
- // reason we did the clustering of layers per thread.
- Polygons filled_polyons_on_lower_layers;
- double bottom_z = layer->print_z - (bridge_height) - EPSILON;
- if (job_idx > 0) {
- for (int lower_job_idx = job_idx - 1; lower_job_idx >= 0; lower_job_idx--) {
- size_t lower_layer_idx = clustered_layers_for_threads[cluster_idx][lower_job_idx];
- const Layer *lower_layer = po->get_layer(lower_layer_idx);
- if (lower_layer->print_z >= bottom_z) {
- for (const auto &c : surfaces_by_layer[lower_layer_idx]) {
- filled_polyons_on_lower_layers.insert(filled_polyons_on_lower_layers.end(), c.new_polys.begin(),
- c.new_polys.end());
- }
- } else {
- break;
- }
- }
- }
- deep_infill_area = diff(deep_infill_area, filled_polyons_on_lower_layers);
- }
- deep_infill_area = expand(deep_infill_area, spacing * 1.5);
- // Now gather expansion polygons - internal infill on current layer, from which we can cut off anchors
- Polygons lightning_area;
- Polygons expansion_area;
- Polygons total_fill_area;
- for (const LayerRegion *region : layer->regions()) {
- Polygons internal_polys = to_polygons(region->fill_surfaces().filter_by_types({(stPosInternal | stDensSparse), (stPosInternal | stDensSolid)}));
- expansion_area.insert(expansion_area.end(), internal_polys.begin(), internal_polys.end());
- Polygons fill_polys = to_polygons(region->fill_expolygons());
- total_fill_area.insert(total_fill_area.end(), fill_polys.begin(), fill_polys.end());
- if (region->region().config().fill_pattern.value == ipLightning) {
- Polygons l = to_polygons(region->fill_surfaces().filter_by_type((stPosInternal | stDensSparse)));
- lightning_area.insert(lightning_area.end(), l.begin(), l.end());
- }
- }
- total_fill_area = closing(total_fill_area, float(SCALED_EPSILON));
- expansion_area = closing(expansion_area, float(SCALED_EPSILON));
- expansion_area = intersection(expansion_area, deep_infill_area);
- Polylines anchors = intersection_pl(infill_lines[lidx - 1], shrink(expansion_area, spacing));
- Polygons internal_unsupported_area = shrink(deep_infill_area, spacing * 4.5);
- #ifdef DEBUG_BRIDGE_OVER_INFILL
- debug_draw(std::to_string(lidx) + "_" + std::to_string(cluster_idx) + "_" + std::to_string(job_idx) + "_" + "_total_area",
- to_lines(total_fill_area), to_lines(expansion_area), to_lines(deep_infill_area), to_lines(anchors));
- #endif
- std::vector<CandidateSurface> expanded_surfaces;
- expanded_surfaces.reserve(surfaces_by_layer[lidx].size());
- for (const CandidateSurface &candidate : surfaces_by_layer[lidx]) {
- const Flow &flow = candidate.region->bridging_flow(frSolidInfill, candidate.region->region().config().bridge_type);
- Polygons area_to_be_bridge = expand(candidate.new_polys, flow.scaled_spacing());
- area_to_be_bridge = intersection(area_to_be_bridge, deep_infill_area);
- area_to_be_bridge.erase(std::remove_if(area_to_be_bridge.begin(), area_to_be_bridge.end(),
- [internal_unsupported_area](const Polygon &p) {
- return intersection({p}, internal_unsupported_area).empty();
- }),
- area_to_be_bridge.end());
- Polygons limiting_area = union_(area_to_be_bridge, expansion_area);
- if (area_to_be_bridge.empty())
- continue;
- Polylines boundary_plines = to_polylines(expand(total_fill_area, 1.3 * flow.scaled_spacing()));
- {
- Polylines limiting_plines = to_polylines(expand(limiting_area, 0.3*flow.spacing()));
- boundary_plines.insert(boundary_plines.end(), limiting_plines.begin(), limiting_plines.end());
- }
- #ifdef DEBUG_BRIDGE_OVER_INFILL
- int r = rand();
- debug_draw(std::to_string(lidx) + "_" + std::to_string(cluster_idx) + "_" + std::to_string(job_idx) + "_" +
- "_anchors_" + std::to_string(r),
- to_lines(area_to_be_bridge), to_lines(boundary_plines), to_lines(anchors), to_lines(expansion_area));
- #endif
- double bridging_angle = 0;
- if (!anchors.empty()) {
- bridging_angle = determine_bridging_angle(area_to_be_bridge, to_lines(anchors),
- candidate.region->region().config().fill_pattern.value);
- } else {
- // use expansion boundaries as anchors.
- // Also, use Infill pattern that is neutral for angle determination, since there are no infill lines.
- bridging_angle = determine_bridging_angle(area_to_be_bridge, to_lines(boundary_plines), InfillPattern::ipLine);
- }
- boundary_plines.insert(boundary_plines.end(), anchors.begin(), anchors.end());
- if (!lightning_area.empty() && !intersection(area_to_be_bridge, lightning_area).empty()) {
- boundary_plines = intersection_pl(boundary_plines, expand(area_to_be_bridge, scale_(10)));
- }
- Polygons bridging_area = construct_anchored_polygon(area_to_be_bridge, to_lines(boundary_plines), flow, bridging_angle);
- // Check collision with other expanded surfaces
- {
- bool reconstruct = false;
- Polygons tmp_expanded_area = expand(bridging_area, 3.0 * flow.scaled_spacing());
- for (const CandidateSurface &s : expanded_surfaces) {
- if (!intersection(s.new_polys, tmp_expanded_area).empty()) {
- bridging_angle = s.bridge_angle;
- reconstruct = true;
- break;
- }
- }
- if (reconstruct) {
- bridging_area = construct_anchored_polygon(area_to_be_bridge, to_lines(boundary_plines), flow, bridging_angle);
- }
- }
- bridging_area = opening(bridging_area, flow.scaled_spacing());
- bridging_area = closing(bridging_area, flow.scaled_spacing());
- bridging_area = intersection(bridging_area, limiting_area);
- bridging_area = intersection(bridging_area, total_fill_area);
- expansion_area = diff(expansion_area, bridging_area);
- #ifdef DEBUG_BRIDGE_OVER_INFILL
- debug_draw(std::to_string(lidx) + "_" + std::to_string(cluster_idx) + "_" + std::to_string(job_idx) + "_" + "_expanded_bridging" + std::to_string(r),
- to_lines(layer->lslices()), to_lines(boundary_plines), to_lines(candidate.new_polys), to_lines(bridging_area));
- #endif
- expanded_surfaces.push_back(CandidateSurface(candidate.original_surface, candidate.layer_index, bridging_area,
- candidate.region, bridging_angle));
- }
- surfaces_by_layer[lidx].swap(expanded_surfaces);
- expanded_surfaces.clear();
- }
- }
- );
- BOOST_LOG_TRIVIAL(info) << "Bridge over infill - Directions and expanded surfaces computed" << log_memory_info();
-
- Slic3r::parallel_for(size_t(0), this->layers().size(),
- [po = this, &surfaces_by_layer]
- (const size_t lidx) {
- coord_t scaled_resolution = std::max(SCALED_EPSILON, scale_t(po->print()->config().resolution.value));
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- if (!(surfaces_by_layer.find(lidx) == surfaces_by_layer.end() && surfaces_by_layer.find(lidx + 1) == surfaces_by_layer.end())) {
- Layer *layer = po->get_layer(lidx);
- Polygons cut_from_infill{};
- if (surfaces_by_layer.find(lidx) != surfaces_by_layer.end()) {
- for (const auto &surface : surfaces_by_layer.at(lidx)) {
- cut_from_infill.insert(cut_from_infill.end(), surface.new_polys.begin(), surface.new_polys.end());
- }
- }
- Polygons additional_ensuring_areas{};
- if (surfaces_by_layer.find(lidx + 1) != surfaces_by_layer.end()) {
- for (const auto &surface : surfaces_by_layer.at(lidx + 1)) {
- auto additional_area = diff(surface.new_polys,
- shrink(surface.new_polys, surface.region->flow(frSolidInfill).scaled_spacing()));
- additional_ensuring_areas.insert(additional_ensuring_areas.end(), additional_area.begin(), additional_area.end());
- }
- }
- for (LayerRegion *region : layer->regions()) {
- Surfaces new_surfaces;
- Polygons near_perimeters = to_polygons(union_safety_offset_ex(to_polygons(region->fill_surfaces().surfaces)));
- near_perimeters = diff(near_perimeters, shrink(near_perimeters, region->flow(frSolidInfill).scaled_spacing()));
- ExPolygons additional_ensuring = intersection_ex(additional_ensuring_areas, near_perimeters);
- SurfacesPtr internal_infills = region->m_fill_surfaces.filter_by_type((stPosInternal | stDensSparse));
- ExPolygons new_internal_infills = diff_ex(internal_infills, cut_from_infill);
- new_internal_infills = diff_ex(new_internal_infills, additional_ensuring);
- ensure_valid(new_internal_infills, scaled_resolution);
- assert_valid(new_internal_infills);
- for (const ExPolygon &ep : new_internal_infills) {
- ep.assert_valid();
- new_surfaces.emplace_back((stPosInternal | stDensSparse), ep);
- }
- SurfacesPtr internal_solids = region->m_fill_surfaces.filter_by_type((stPosInternal | stDensSolid));
- if (surfaces_by_layer.find(lidx) != surfaces_by_layer.end()) {
- for (const CandidateSurface &cs : surfaces_by_layer.at(lidx)) {
- for (const Surface *surface : internal_solids) {
- if (cs.original_surface == surface) {
- Surface tmp{*surface, {}};
- tmp.surface_type = (stPosInternal | stDensSolid | stModBridge);
- tmp.bridge_angle = cs.bridge_angle;
- for (const ExPolygon &ep : ensure_valid(union_ex(cs.new_polys), scaled_resolution)) {
- ep.assert_valid();
- new_surfaces.emplace_back(tmp, ep);
- }
- break;
- }
- }
- }
- }
- ExPolygons new_internal_solids = to_expolygons(internal_solids);
- new_internal_solids.insert(new_internal_solids.end(), additional_ensuring.begin(), additional_ensuring.end());
- new_internal_solids = diff_ex(new_internal_solids, cut_from_infill);
- new_internal_solids = union_safety_offset_ex(new_internal_solids);
- ensure_valid(new_internal_solids, scaled_resolution);
- for (const ExPolygon &ep : new_internal_solids) {
- ep.assert_valid();
- new_surfaces.emplace_back((stPosInternal | stDensSolid), ep);
- }
- #ifdef DEBUG_BRIDGE_OVER_INFILL
- debug_draw("Aensuring_" + std::to_string(reinterpret_cast<uint64_t>(®ion)), to_polylines(additional_ensuring),
- to_polylines(near_perimeters), to_polylines(to_polygons(internal_infills)),
- to_polylines(to_polygons(internal_solids)));
- debug_draw("Aensuring_" + std::to_string(reinterpret_cast<uint64_t>(®ion)) + "_new", to_polylines(additional_ensuring),
- to_polylines(near_perimeters), to_polylines(to_polygons(new_internal_infills)),
- to_polylines(to_polygons(new_internal_solids)));
- #endif
- region->m_fill_surfaces.remove_types({(stPosInternal | stDensSolid), (stPosInternal | stDensSparse)});
- for(auto &srf : new_surfaces)
- srf.expolygon.assert_valid();
- region->m_fill_surfaces.append(new_surfaces);
- }
- }
- });
- BOOST_LOG_TRIVIAL(info) << "Bridge over infill - End" << log_memory_info();
- } // void PrintObject::bridge_over_infill()
- static void clamp_exturder_to_default(ConfigOptionInt &opt, size_t num_extruders)
- {
- if (opt.value > (int)num_extruders)
- // assign the default extruder
- opt.value = 1;
- }
- PrintObjectConfig PrintObject::object_config_from_model_object(const PrintObjectConfig &default_object_config, const ModelObject &object, size_t num_extruders)
- {
- PrintObjectConfig config = default_object_config;
- {
- DynamicPrintConfig src_normalized(object.config.get());
- src_normalized.normalize_fdm();
- config.apply(src_normalized, true);
- }
- // Clamp invalid extruders to the default extruder (with index 1).
- clamp_exturder_to_default(config.support_material_extruder, num_extruders);
- clamp_exturder_to_default(config.support_material_interface_extruder, num_extruders);
- return config;
- }
- const std::string key_extruder { "extruder" };
- static constexpr const std::initializer_list<const std::string_view> keys_extruders { "infill_extruder"sv, "solid_infill_extruder"sv, "perimeter_extruder"sv };
- static void apply_to_print_region_config(PrintRegionConfig &out, const DynamicPrintConfig &in)
- {
- // 1) Copy the "extruder key to infill_extruder and perimeter_extruder.
- auto *opt_extruder = in.opt<ConfigOptionInt>(key_extruder);
- if (opt_extruder)
- if (int extruder = opt_extruder->value; extruder != 0) {
- // Not a default extruder.
- out.infill_extruder .value = extruder;
- out.solid_infill_extruder.value = extruder;
- out.perimeter_extruder .value = extruder;
- }
- // 2) Copy the rest of the values.
- for (auto it = in.cbegin(); it != in.cend(); ++ it)
- if (it->first != key_extruder)
- if (ConfigOption* my_opt = out.option(it->first, false); my_opt != nullptr) {
- if (one_of(it->first, keys_extruders)) {
- assert(dynamic_cast<ConfigOptionInt*>(my_opt));
- // Ignore "default" extruders.
- int extruder = static_cast<const ConfigOptionInt*>(it->second.get())->value;
- if (extruder > 0)
- static_cast<ConfigOptionInt *>(my_opt)->value = (extruder);
- } else
- my_opt->set(it->second.get());
- }
- }
- PrintRegionConfig region_config_from_model_volume(const PrintRegionConfig &default_or_parent_region_config, const DynamicPrintConfig *layer_range_config, const ModelVolume &volume, size_t num_extruders)
- {
- PrintRegionConfig config = default_or_parent_region_config;
- if (volume.is_model_part()) {
- // default_or_parent_region_config contains the Print's PrintRegionConfig.
- // Override with ModelObject's PrintRegionConfig values.
- apply_to_print_region_config(config, volume.get_object()->config.get());
- } else {
- // default_or_parent_region_config contains parent PrintRegion config, which already contains ModelVolume's config.
- }
- if (layer_range_config != nullptr) {
- // Not applicable to modifiers.
- assert(volume.is_model_part());
- apply_to_print_region_config(config, *layer_range_config);
- }
- apply_to_print_region_config(config, volume.config.get());
- if (! volume.material_id().empty())
- apply_to_print_region_config(config, volume.material()->config.get());
- // Clamp invalid extruders to the default extruder (with index 1).
- clamp_exturder_to_default(config.infill_extruder, num_extruders);
- clamp_exturder_to_default(config.perimeter_extruder, num_extruders);
- clamp_exturder_to_default(config.solid_infill_extruder, num_extruders);
- if (config.fill_density.value < 0.00011f)
- // Switch of infill for very low infill rates, also avoid division by zero in infill generator for these very low rates.
- // See GH issue #5910.
- config.fill_density.value = 0;
- else
- config.fill_density.value = std::min(config.fill_density.value, 100.);
- if (config.fuzzy_skin.value != FuzzySkinType::None && (config.fuzzy_skin_point_dist.value < 0.01 || config.fuzzy_skin_thickness.value < 0.001))
- config.fuzzy_skin.value = FuzzySkinType::None;
- return config;
- }
- void PrintObject::update_slicing_parameters()
- {
- if (!m_slicing_params || !m_slicing_params->valid)
- m_slicing_params = SlicingParameters::create_from_config(
- this->print()->config(), m_config, this->print()->default_region_config(), this->model_object()->max_z()/*bounding_box().max.z()*/, this->object_extruders());
- }
- std::shared_ptr<SlicingParameters> PrintObject::slicing_parameters(const DynamicPrintConfig& full_config, const ModelObject& model_object, float object_max_z)
- {
- PrintConfig print_config;
- PrintObjectConfig object_config;
- PrintRegionConfig default_region_config;
- print_config.apply(full_config, true);
- object_config.apply(full_config, true);
- default_region_config.apply(full_config, true);
- size_t num_extruders = print_config.nozzle_diameter.size();
- object_config = object_config_from_model_object(object_config, model_object, num_extruders);
- std::set<uint16_t> object_extruders;
- for (const ModelVolume* model_volume : model_object.volumes)
- if (model_volume->is_model_part()) {
- PrintRegion::collect_object_printing_extruders(
- print_config,
- object_config,
- region_config_from_model_volume(default_region_config, nullptr, *model_volume, num_extruders),
- object_extruders);
- for (const std::pair<const t_layer_height_range, ModelConfig>& range_and_config : model_object.layer_config_ranges)
- if (range_and_config.second.has("perimeter_extruder") ||
- range_and_config.second.has("infill_extruder") ||
- range_and_config.second.has("solid_infill_extruder"))
- PrintRegion::collect_object_printing_extruders(
- print_config,
- object_config,
- region_config_from_model_volume(default_region_config, &range_and_config.second.get(), *model_volume, num_extruders),
- object_extruders);
- }
- //FIXME add painting extruders
- if (object_max_z <= 0.f)
- object_max_z = (float)model_object.raw_bounding_box().size().z();
- return SlicingParameters::create_from_config(print_config, object_config, default_region_config, object_max_z, object_extruders);
- }
- // returns 0-based indices of extruders used to print the object (without brim, support and other helper extrusions)
- std::set<uint16_t> PrintObject::object_extruders() const
- {
- std::set<uint16_t> extruders;
- for (const PrintRegion ®ion : this->all_regions())
- region.collect_object_printing_extruders(*this->print(), extruders);
- return extruders;
- }
- double PrintObject::get_first_layer_height() const
- {
- //get object first layer height
- double object_first_layer_height = config().first_layer_height.value;
- if (config().first_layer_height.percent) {
- object_first_layer_height = 1000000000;
- for (uint16_t extruder_id : object_extruders()) {
- double nozzle_diameter = print()->config().nozzle_diameter.get_at(extruder_id);
- object_first_layer_height = std::fmin(object_first_layer_height, config().first_layer_height.get_abs_value(nozzle_diameter));
- }
- }
- assert(object_first_layer_height < 1000000000);
- return object_first_layer_height;
- }
- bool PrintObject::update_layer_height_profile(const ModelObject& model_object, const SlicingParameters& slicing_parameters, std::vector<coordf_t>& layer_height_profile)
- {
- bool updated = false;
- if (layer_height_profile.empty()) {
- // use the constructor because the assignement is crashing on ASAN OsX
- layer_height_profile = model_object.layer_height_profile.get();
- // layer_height_profile = model_object.layer_height_profile;
- // The layer height returned is sampled with high density for the UI layer height painting
- // and smoothing tool to work.
- updated = true;
- }
- // Verify the layer_height_profile.
- if (!layer_height_profile.empty() &&
- // Must not be of even length.
- ((layer_height_profile.size() & 1) != 0 ||
- // Last entry must be at the top of the object.
- std::abs(layer_height_profile[layer_height_profile.size() - 2] - slicing_parameters.object_print_z_max + slicing_parameters.object_print_z_min) > 10 * EPSILON)) {
- if ((layer_height_profile.size() & 1) != 0) {
- BOOST_LOG_TRIVIAL(error) << "Error: can't apply the layer hight profile: layer_height_profile array is odd, not even.";
- } else {
- BOOST_LOG_TRIVIAL(error) << "Error: can't apply the layer hight profile: layer_height_profile last layer is at "
- << layer_height_profile[layer_height_profile.size() - 2]
- <<", and it's too far away from object_print_z_max = "<<(slicing_parameters.object_print_z_max + slicing_parameters.object_print_z_min);
- }
- layer_height_profile.clear();
- }
- if (layer_height_profile.empty()) {
- //layer_height_profile = layer_height_profile_adaptive(slicing_parameters, model_object.layer_config_ranges, model_object.volumes);
- layer_height_profile = layer_height_profile_from_ranges(slicing_parameters, model_object.layer_config_ranges);
- // The layer height profile is already compressed.
- updated = true;
- }
- return updated;
- }
- // Only active if config->infill_only_where_needed. This step trims the sparse infill,
- // so it acts as an internal support. It maintains all other infill types intact.
- // Here the internal surfaces and perimeters have to be supported by the sparse infill.
- //FIXME The surfaces are supported by a sparse infill, but the sparse infill is only as large as the area to support.
- // Likely the sparse infill will not be anchored correctly, so it will not work as intended.
- // Also one wishes the perimeters to be supported by a full infill.
- // Idempotence of this method is guaranteed by the fact that we don't remove things from
- // fill_surfaces but we only turn them into VOID surfaces, thus preserving the boundaries.
- // void PrintObject::clip_fill_surfaces()
- // {
- // bool has_lightning_infill = false;
- // for (size_t region_id = 0; region_id < this->num_printing_regions(); ++region_id)
- // if (const PrintRegionConfig &config = this->printing_region(region_id).config(); config.fill_density > 0 && config.fill_pattern.value == ipLightning)
- // has_lightning_infill = true;
- // // For Lightning infill, infill_only_where_needed is ignored because both
- // // do a similar thing, and their combination doesn't make much sense.
- // if (! m_config.infill_only_where_needed.value || has_lightning_infill)
- // return;
- // bool has_infill = false;
- // for (size_t i = 0; i < this->num_printing_regions(); ++ i)
- // if (this->printing_region(i).config().fill_density > 0) {
- // has_infill = true;
- // break;
- // }
- // if (! has_infill)
- // return;
- // // We only want infill under ceilings; this is almost like an
- // // internal support material.
- // // Proceed top-down, skipping the bottom layer.
- // Polygons upper_internal;
- // for (int layer_id = int(m_layers.size()) - 1; layer_id > 0; -- layer_id) {
- // Layer *layer = m_layers[layer_id];
- // Layer *lower_layer = m_layers[layer_id - 1];
- // // Detect things that we need to support.
- // // Cummulative fill surfaces.
- // Polygons fill_surfaces;
- // // Solid surfaces to be supported.
- // Polygons overhangs;
- // for (const LayerRegion *layerm : layer->m_regions)
- // for (const Surface &surface : layerm->fill_surfaces()) {
- // Polygons polygons = to_polygons(surface.expolygon);
- // if (surface.has_fill_solid())
- // polygons_append(overhangs, polygons);
- // polygons_append(fill_surfaces, std::move(polygons));
- // }
- // Polygons lower_layer_fill_surfaces;
- // Polygons lower_layer_internal_surfaces;
- // for (const LayerRegion *layerm : lower_layer->m_regions)
- // for (const Surface &surface : layerm->fill_surfaces()) {
- // Polygons polygons = to_polygons(surface.expolygon);
- // if (surface.has_pos_internal() && (surface.has_fill_sparse() || surface.has_fill_void()))
- // polygons_append(lower_layer_internal_surfaces, polygons);
- // polygons_append(lower_layer_fill_surfaces, std::move(polygons));
- // }
- // // We also need to support perimeters when there's at least one full unsupported loop
- // {
- // // Get perimeters area as the difference between slices and fill_surfaces
- // // Only consider the area that is not supported by lower perimeters
- // Polygons perimeters = intersection(diff(layer->lslices(), fill_surfaces), lower_layer_fill_surfaces);
- // // Only consider perimeter areas that are at least one extrusion width thick.
- // //FIXME Offset2 eats out from both sides, while the perimeters are create outside in.
- // //Should the pw not be half of the current value?
- // float pw = FLT_MAX;
- // for (const LayerRegion *layerm : layer->m_regions)
- // pw = std::min(pw, (float)layerm->flow(frPerimeter).scaled_width());
- // // Append such thick perimeters to the areas that need support
- // polygons_append(overhangs, opening(perimeters, pw));
- // }
- // // Merge the new overhangs, find new internal infill.
- // polygons_append(upper_internal, std::move(overhangs));
- // static constexpr const auto closing_radius = scaled<float>(2.f);
- // upper_internal = intersection(
- // // Regularize the overhang regions, so that the infill areas will not become excessively jagged.
- // smooth_outward(
- // closing(upper_internal, closing_radius, ClipperLib::jtSquare, 0.),
- // scaled<coord_t>(0.1)),
- // lower_layer_internal_surfaces);
- // // Apply new internal infill to regions.
- // for (LayerRegion *layerm : lower_layer->m_regions) {
- // if (layerm->region().config().fill_density.value == 0 || layerm->region().config().infill_dense.value)
- // continue;
- // Polygons internal;
- // for (Surface &surface : layerm->m_fill_surfaces.surfaces)
- // if (surface.surface_type == (stPosInternal | stDensSparse) || surface.surface_type == stInternalVoid)
- // polygons_append(internal, std::move(surface.expolygon));
- // layerm->m_fill_surfaces.remove_types({ stPosInternal | stDensSparse, stPosInternal | stDensVoid });
- // layerm->m_fill_surfaces.append(intersection_ex(internal, upper_internal, ApplySafetyOffset::Yes), stPosInternal | stDensSparse);
- // layerm->m_fill_surfaces.append(diff_ex (internal, upper_internal, ApplySafetyOffset::Yes), stPosInternal | stDensVoid);
- // // If there are voids it means that our internal infill is not adjacent to
- // // perimeters. In this case it would be nice to add a loop around infill to
- // // make it more robust and nicer. TODO.
- // #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- // layerm->export_region_fill_surfaces_to_svg_debug("6_clip_fill_surfaces");
- // #endif
- // }
- // m_print->throw_if_canceled();
- // }
- // } // void PrintObject::clip_fill_surfaces()
- void PrintObject::discover_horizontal_shells()
- {
- BOOST_LOG_TRIVIAL(trace) << "discover_horizontal_shells()";
- for (size_t region_id = 0; region_id < this->num_printing_regions(); ++region_id) {
- for (size_t i = 0; i < m_layers.size(); ++i) {
- m_print->throw_if_canceled();
- Layer *layer = m_layers[i];
- LayerRegion *layerm = layer->get_region(region_id);
- const PrintRegionConfig ®ion_config = layerm->region().config();
- if (region_config.solid_infill_every_layers.value > 0 && region_config.fill_density.value > 0 &&
- (i % region_config.solid_infill_every_layers) == 0) {
- // Insert a solid internal layer. Mark stInternal surfaces as stInternalSolid or stInternalBridge.
- SurfaceType type = (region_config.fill_density == 100 || region_config.solid_infill_every_layers == 1) ?
- (stPosInternal | stDensSolid) :
- (stPosInternal | stDensSolid | stModBridge);
- for (Surface& surface : layerm->m_fill_surfaces.surfaces)
- if (surface.surface_type == (stPosInternal | stDensSparse))
- surface.surface_type = type;
- }
- // The rest has already been performed by discover_vertical_shells().
- } // for each layer
- } // for each region
- #ifdef SLIC3R_DEBUG_SLICE_PROCESSING
- for (size_t region_id = 0; region_id < this->num_printing_regions(); ++region_id) {
- for (const Layer *layer : m_layers) {
- const LayerRegion *layerm = layer->m_regions[region_id];
- layerm->export_region_slices_to_svg_debug("5_discover_horizontal_shells");
- layerm->export_region_fill_surfaces_to_svg_debug("5_discover_horizontal_shells");
- } // for each layer
- } // for each region
- #endif /* SLIC3R_DEBUG_SLICE_PROCESSING */
- } // void PrintObject::discover_horizontal_shells()
- void merge_surfaces(LayerRegion* lregion) {
- coord_t scaled_resolution = std::max(SCALED_EPSILON, scale_t(lregion->layer()->object()->print()->config().resolution.value));
- //merge regions with same type (other things are all the same anyway)
- std::map< SurfaceType, std::vector<const Surface*>> type2srfs;
- for (const Surface& surface : lregion->fill_surfaces().surfaces) {
- type2srfs[surface.surface_type].push_back(&surface);
- }
- bool changed = false;
- std::map< SurfaceType, ExPolygons> type2newpolys;
- for (auto& entry : type2srfs) {
- if (entry.second.size() > 2) {
- ExPolygons merged = ensure_valid(union_safety_offset_ex(to_expolygons(entry.second)), scaled_resolution);
- if (merged.size() < entry.second.size()) {
- changed = true;
- type2newpolys[entry.first] = std::move(merged);
- }
- }
- }
- if (changed) {
- Surfaces newSrfs;
- for (auto& entry : type2srfs) {
- if (type2newpolys.find(entry.first) == type2newpolys.end()) {
- for (const Surface* srfPtr : entry.second) {
- srfPtr->expolygon.assert_valid();
- newSrfs.emplace_back(*srfPtr);
- }
- } else {
- for (ExPolygon& expoly : type2newpolys[entry.first]) {
- expoly.assert_valid();
- newSrfs.emplace_back(*entry.second.front(), expoly);
- }
- }
- }
- lregion->set_fill_surfaces().surfaces = std::move(newSrfs);
- }
- }
- void PrintObject::clean_surfaces() {
- Slic3r::parallel_for(size_t(0), this->layers().size() - 1,
- [this](const size_t idx_layer) {
- for (LayerRegion* lregion : this->layers()[idx_layer]->regions()) {
- coord_t extrusion_width = lregion->flow(frInfill).scaled_width();
- merge_surfaces(lregion);
- // collapse too thin solid surfaces.
- bool changed_type = false;
- for (Surface& surface : lregion->set_fill_surfaces().surfaces) {
- if (surface.has_fill_solid() && surface.has_pos_internal()) {
- if (offset2_ex(ExPolygons{ surface.expolygon }, -extrusion_width / 2, extrusion_width / 2).empty()) {
- //convert to sparse
- surface.surface_type = (surface.surface_type ^ SurfaceType::stDensSolid) | SurfaceType::stDensSparse;
- changed_type = true;
- }
- }
- }
- merge_surfaces(lregion);
- }
- }
- );
- }
- // combine fill surfaces across layers to honor the "infill every N layers" option
- // Idempotence of this method is guaranteed by the fact that we don't remove things from
- // fill_surfaces but we only turn them into VOID surfaces, thus preserving the boundaries.
- void PrintObject::combine_infill()
- {
- // Work on each region separately.
- for (size_t region_id = 0; region_id < this->num_printing_regions(); ++ region_id) {
- const PrintRegion ®ion = this->printing_region(region_id);
- // can't have void if using infill_dense
- const size_t every = region.config().infill_dense.value ? 1 : region.config().infill_every_layers.value;
- if (every < 2 || region.config().fill_density == 0.)
- continue;
- // Limit the number of combined layers to the maximum height allowed by this regions' nozzle.
- //FIXME limit the layer height to max_layer_height
- double nozzle_diameter = std::min(
- this->print()->config().nozzle_diameter.get_at(region.config().infill_extruder.value - 1),
- this->print()->config().nozzle_diameter.get_at(region.config().solid_infill_extruder.value - 1));
- // define the combinations
- std::vector<size_t> combine(m_layers.size(), 0);
- {
- double current_height = 0.;
- size_t num_layers = 0;
- for (size_t layer_idx = 0; layer_idx < m_layers.size(); ++ layer_idx) {
- m_print->throw_if_canceled();
- const Layer *layer = m_layers[layer_idx];
- if (layer->id() == 0)
- // Skip first print layer (which may not be first layer in array because of raft).
- continue;
- // Check whether the combination of this layer with the lower layers' buffer
- // would exceed max layer height or max combined layer count.
- if (current_height + layer->height >= nozzle_diameter + EPSILON || num_layers >= every) {
- // Append combination to lower layer.
- combine[layer_idx - 1] = num_layers;
- current_height = 0.;
- num_layers = 0;
- }
- current_height += layer->height;
- ++ num_layers;
- }
-
- // Append lower layers (if any) to uppermost layer.
- combine[m_layers.size() - 1] = num_layers;
- }
-
- // loop through layers to which we have assigned layers to combine
- for (size_t layer_idx = 0; layer_idx < m_layers.size(); ++ layer_idx) {
- m_print->throw_if_canceled();
- size_t num_layers = combine[layer_idx];
- if (num_layers <= 1)
- continue;
- // Get all the LayerRegion objects to be combined.
- std::vector<LayerRegion*> layerms;
- layerms.reserve(num_layers);
- for (size_t i = layer_idx + 1 - num_layers; i <= layer_idx; ++ i)
- layerms.emplace_back(m_layers[i]->regions()[region_id]);
- // We need to perform a multi-layer intersection, so let's split it in pairs.
- // Initialize the intersection with the candidates of the lowest layer.
- ExPolygons intersection = to_expolygons(layerms.front()->fill_surfaces().filter_by_type(stPosInternal | stDensSparse));
- // Start looping from the second layer and intersect the current intersection with it.
- for (size_t i = 1; i < layerms.size(); ++i)
- intersection = intersection_ex(to_expolygons(layerms[i]->fill_surfaces().filter_by_type(stPosInternal | stDensSparse)), intersection);
- double area_threshold = layerms.front()->infill_area_threshold();
- if (! intersection.empty() && area_threshold > 0.)
- intersection.erase(std::remove_if(intersection.begin(), intersection.end(),
- [area_threshold](const ExPolygon &expoly) { return expoly.area() <= area_threshold; }),
- intersection.end());
- if (intersection.empty())
- continue;
- // Slic3r::debugf " combining %d %s regions from layers %d-%d\n",
- // scalar(@$intersection),
- // ($type == stInternal ? 'internal' : 'internal-solid'),
- // $layer_idx-($every-1), $layer_idx;
- // intersection now contains the regions that can be combined across the full amount of layers,
- // so let's remove those areas from all layers.
- Polygons intersection_with_clearance;
- intersection_with_clearance.reserve(intersection.size());
- //TODO: check if that 'hack' isn't counter-productive : the overlap is done at perimetergenerator (so before this)
- // and the not-overlap area is stored in the LayerRegion object
- float clearance_offset =
- 0.5f * layerms.back()->flow(frPerimeter).scaled_width() +
- // Because fill areas for rectilinear and honeycomb are grown
- // later to overlap perimeters, we need to counteract that too.
- ((region.config().fill_pattern.value == ipRectilinear ||
- region.config().fill_pattern.value == ipMonotonic ||
- region.config().fill_pattern.value == ipGrid ||
- region.config().fill_pattern.value == ipLine ||
- region.config().fill_pattern.value == ipHoneycomb) ? 1.5f : 0.5f) *
- layerms.back()->flow(frSolidInfill).scaled_width();
- for (ExPolygon& expoly : intersection) {
- expoly.assert_valid();
- polygons_append(intersection_with_clearance, offset(expoly, clearance_offset));
- }
- for (LayerRegion *layerm : layerms) {
- Polygons internal = to_polygons(layerm->fill_surfaces().filter_by_type(stPosInternal | stDensSparse));
- layerm->m_fill_surfaces.remove_type(stPosInternal | stDensSparse);
- ExPolygons only_internal = diff_ex(internal, intersection_with_clearance);
- assert_valid(only_internal);
- layerm->m_fill_surfaces.append(std::move(only_internal), stPosInternal | stDensSparse);
- if (layerm == layerms.back()) {
- // Apply surfaces back with adjusted depth to the uppermost layer.
- Surface templ(stPosInternal | stDensSparse, ExPolygon());
- templ.thickness = 0.;
- for (LayerRegion* layerm2 : layerms) {
- templ.thickness += layerm2->layer()->height;
- }
- templ.thickness_layers = (unsigned short)layerms.size();
- layerm->m_fill_surfaces.append(intersection, templ);
- } else {
- // Save void surfaces.
- ExPolygons only_void = intersection_ex(internal, intersection_with_clearance);
- assert_valid(only_void);
- layerm->m_fill_surfaces.append(std::move(only_void), stPosInternal | stDensVoid);
- }
- }
- }
- }
- } // void PrintObject::combine_infill()
- void PrintObject::_generate_support_material()
- {
- if (this->has_support() && (m_config.support_material_style.value == smsTree || m_config.support_material_style.value == smsOrganic)) {
- fff_tree_support_generate(*this, std::function<void()>([this](){ this->throw_if_canceled(); }));
- } else {
- // If support style is set to Organic however only raft will be built but no support,
- // build snug raft instead.
- PrintObjectSupportMaterial support_material(this, m_slicing_params);
- support_material.generate(*this);
- }
- }
- static void project_triangles_to_slabs(SpanOfConstPtrs<Layer> layers, const indexed_triangle_set &custom_facets, const Transform3f &tr, bool seam, std::vector<Polygons> &out)
- {
- if (custom_facets.indices.empty())
- return;
- const float tr_det_sign = (tr.matrix().determinant() > 0. ? 1.f : -1.f);
- // The projection will be at most a pentagon. Let's minimize heap
- // reallocations by saving in in the following struct.
- // Points are used so that scaling can be done in parallel
- // and they can be moved from to create an ExPolygon later.
- struct LightPolygon {
- LightPolygon() { pts.reserve(5); }
- LightPolygon(const std::array<Vec2f, 3>& tri) {
- pts.reserve(3);
- pts.emplace_back(scaled<coord_t>(tri.front()));
- pts.emplace_back(scaled<coord_t>(tri[1]));
- pts.emplace_back(scaled<coord_t>(tri.back()));
- }
- Points pts;
- void add(const Vec2f& pt) {
- pts.emplace_back(scaled<coord_t>(pt));
- assert(pts.size() <= 5);
- }
- };
- // Structure to collect projected polygons. One element for each triangle.
- // Saves vector of polygons and layer_id of the first one.
- struct TriangleProjections {
- size_t first_layer_id;
- std::vector<LightPolygon> polygons;
- };
- // Vector to collect resulting projections from each triangle.
- std::vector<TriangleProjections> projections_of_triangles(custom_facets.indices.size());
- // Iterate over all triangles.
- Slic3r::parallel_for(size_t(0), custom_facets.indices.size(),
- [&custom_facets, &tr, tr_det_sign, seam, layers, &projections_of_triangles]
- (const size_t idx) {
- PRINT_OBJECT_TIME_LIMIT_MILLIS(PRINT_OBJECT_TIME_LIMIT_DEFAULT);
- std::array<Vec3f, 3> facet;
- // Transform the triangle into worlds coords.
- for (int i=0; i<3; ++i)
- facet[i] = tr * custom_facets.vertices[custom_facets.indices[idx](i)];
- // Ignore triangles with upward-pointing normal. Don't forget about mirroring.
- float z_comp = (facet[1]-facet[0]).cross(facet[2]-facet[0]).z();
- if (! seam && tr_det_sign * z_comp > 0.)
- return; //continue (next facet idx)
- // The algorithm does not process vertical triangles, but it should for seam.
- // In that case, tilt the triangle a bit so the projection does not degenerate.
- if (seam && z_comp == 0.f)
- facet[0].x() += float(EPSILON);
- // Sort the three vertices according to z-coordinate.
- std::sort(facet.begin(), facet.end(),
- [](const Vec3f& pt1, const Vec3f&pt2) {
- return pt1.z() < pt2.z();
- });
- std::array<Vec2f, 3> trianglef;
- for (int i=0; i<3; ++i)
- trianglef[i] = to_2d(facet[i]);
- // Find lowest slice not below the triangle.
- auto it = std::lower_bound(layers.begin(), layers.end(), facet[0].z()+EPSILON,
- [](const Layer* l1, float z) {
- return l1->slice_z < z;
- });
- // Count how many projections will be generated for this triangle
- // and allocate respective amount in projections_of_triangles.
- size_t first_layer_id = projections_of_triangles[idx].first_layer_id = it - layers.begin();
- size_t last_layer_id = first_layer_id;
- // The cast in the condition below is important. The comparison must
- // be an exact opposite of the one lower in the code where
- // the polygons are appended. And that one is on floats.
- while (last_layer_id + 1 < layers.size()
- && float(layers[last_layer_id]->slice_z) <= facet[2].z())
- ++last_layer_id;
- if (first_layer_id == last_layer_id) {
- // The triangle fits just a single slab, just project it. This also avoids division by zero for horizontal triangles.
- float dz = facet[2].z() - facet[0].z();
- assert(dz >= 0);
- // The face is nearly horizontal and it crosses the slicing plane at first_layer_id - 1.
- // Rather add this face to both the planes.
- bool add_below = dz < float(2. * EPSILON) && first_layer_id > 0 && layers[first_layer_id - 1]->slice_z > facet[0].z() - EPSILON;
- projections_of_triangles[idx].polygons.reserve(add_below ? 2 : 1);
- projections_of_triangles[idx].polygons.emplace_back(trianglef);
- if (add_below) {
- -- projections_of_triangles[idx].first_layer_id;
- projections_of_triangles[idx].polygons.emplace_back(trianglef);
- }
- return; //continue (next facet idx)
- }
- projections_of_triangles[idx].polygons.resize(last_layer_id - first_layer_id + 1);
- // Calculate how to move points on triangle sides per unit z increment.
- Vec2f ta(trianglef[1] - trianglef[0]);
- Vec2f tb(trianglef[2] - trianglef[0]);
- ta *= 1.f/(facet[1].z() - facet[0].z());
- tb *= 1.f/(facet[2].z() - facet[0].z());
- // Projection on current slice will be built directly in place.
- LightPolygon* proj = &projections_of_triangles[idx].polygons[0];
- proj->add(trianglef[0]);
- bool passed_first = false;
- bool stop = false;
- // Project a sub-polygon on all slices intersecting the triangle.
- while (it != layers.end()) {
- const float z = float((*it)->slice_z);
- // Projections of triangle sides intersections with slices.
- // a moves along one side, b tracks the other.
- Vec2f a;
- Vec2f b;
- // If the middle vertex was already passed, append the vertex
- // and use ta for tracking the remaining side.
- if (z > facet[1].z() && ! passed_first) {
- proj->add(trianglef[1]);
- ta = trianglef[2]-trianglef[1];
- ta *= 1.f/(facet[2].z() - facet[1].z());
- passed_first = true;
- }
- // This slice is above the triangle already.
- if (z > facet[2].z() || it+1 == layers.end()) {
- proj->add(trianglef[2]);
- stop = true;
- }
- else {
- // Move a, b along the side it currently tracks to get
- // projected intersection with current slice.
- a = passed_first ? (trianglef[1]+ta*(z-facet[1].z()))
- : (trianglef[0]+ta*(z-facet[0].z()));
- b = trianglef[0]+tb*(z-facet[0].z());
- proj->add(a);
- proj->add(b);
- }
- if (stop)
- break;
- // Advance to the next layer.
- ++it;
- ++proj;
- assert(proj <= &projections_of_triangles[idx].polygons.back() );
- // a, b are first two points of the polygon for the next layer.
- proj->add(b);
- proj->add(a);
- }
- }); // end of parallel_for
- // Make sure that the output vector can be used.
- out.resize(layers.size());
- // Now append the collected polygons to respective layers.
- for (auto& trg : projections_of_triangles) {
- int layer_id = int(trg.first_layer_id);
- for (LightPolygon &poly : trg.polygons) {
- if (layer_id >= int(out.size()))
- break; // part of triangle could be projected above top layer
- assert(! poly.pts.empty());
- // The resulting triangles are fed to the Clipper library, which seem to handle flipped triangles well.
- // if (cross2(Vec2d((poly.pts[1] - poly.pts[0]).cast<double>()), Vec2d((poly.pts[2] - poly.pts[1]).cast<double>())) < 0)
- // std::swap(poly.pts.front(), poly.pts.back());
-
- out[layer_id].emplace_back(std::move(poly.pts));
- ++layer_id;
- }
- }
- }
- void PrintObject::project_and_append_custom_facets(
- bool seam, EnforcerBlockerType type, std::vector<Polygons>& out) const
- {
- for (const ModelVolume* mv : this->model_object()->volumes)
- if (mv->is_model_part()) {
- const indexed_triangle_set custom_facets = seam
- ? mv->seam_facets.get_facets_strict(*mv, type)
- : mv->supported_facets.get_facets_strict(*mv, type);
- if (! custom_facets.indices.empty()) {
- if (seam)
- project_triangles_to_slabs(this->layers(), custom_facets,
- (this->trafo_centered() * mv->get_matrix()).cast<float>(),
- seam, out);
- else {
- std::vector<Polygons> projected;
- // Support blockers or enforcers. Project downward facing painted areas upwards to their respective slicing plane.
- slice_mesh_slabs(custom_facets, zs_from_layers(this->layers()), this->trafo_centered() * mv->get_matrix(), nullptr, &projected, [](){});
- // Merge these projections with the output, layer by layer.
- assert(! projected.empty());
- assert(out.empty() || out.size() == projected.size());
- if (out.empty())
- out = std::move(projected);
- else
- for (size_t i = 0; i < out.size(); ++ i)
- append(out[i], std::move(projected[i]));
- }
- }
- }
- }
- const Layer* PrintObject::get_layer_at_printz(coordf_t print_z) const {
- auto it = Slic3r::lower_bound_by_predicate(m_layers.begin(), m_layers.end(), [print_z](const Layer *layer) { return layer->print_z < print_z; });
- return (it == m_layers.end() || (*it)->print_z != print_z) ? nullptr : *it;
- }
- Layer* PrintObject::get_layer_at_printz(coordf_t print_z) { return const_cast<Layer*>(std::as_const(*this).get_layer_at_printz(print_z)); }
- // Get a layer approximately at print_z.
- const Layer* PrintObject::get_layer_at_printz(coordf_t print_z, coordf_t epsilon) const {
- coordf_t limit = print_z - epsilon;
- auto it = Slic3r::lower_bound_by_predicate(m_layers.begin(), m_layers.end(), [limit](const Layer *layer) { return layer->print_z < limit; });
- return (it == m_layers.end() || (*it)->print_z > print_z + epsilon) ? nullptr : *it;
- }
- Layer* PrintObject::get_layer_at_printz(coordf_t print_z, coordf_t epsilon) { return const_cast<Layer*>(std::as_const(*this).get_layer_at_printz(print_z, epsilon)); }
- const Layer *PrintObject::get_first_layer_bellow_printz(coordf_t print_z, coordf_t epsilon) const
- {
- coordf_t limit = print_z + epsilon;
- auto it = Slic3r::lower_bound_by_predicate(m_layers.begin(), m_layers.end(), [limit](const Layer *layer) { return layer->print_z < limit; });
- return (it == m_layers.begin()) ? nullptr : *(--it);
- }
- } // namespace Slic3r
|