forked from Ultimaker/CuraEngine
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathWallsComputation.cpp
151 lines (139 loc) · 5.84 KB
/
WallsComputation.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
//Copyright (c) 2018 Ultimaker B.V.
//CuraEngine is released under the terms of the AGPLv3 or higher.
#include "ExtruderTrain.h"
#include "sliceDataStorage.h"
#include "WallsComputation.h"
#include "settings/types/Ratio.h"
#include "utils/polygonUtils.h"
namespace cura {
WallsComputation::WallsComputation(const Settings& settings, const LayerIndex layer_nr)
: settings(settings)
, layer_nr(layer_nr)
{
}
/*
* This function is executed in a parallel region based on layer_nr.
* When modifying make sure any changes does not introduce data races.
*
* generateInsets only reads and writes data for the current layer
*/
void WallsComputation::generateInsets(SliceLayerPart* part)
{
size_t inset_count = settings.get<size_t>("wall_line_count");
const bool spiralize = settings.get<bool>("magic_spiralize");
if (spiralize && layer_nr < LayerIndex(settings.get<size_t>("bottom_layers")) && ((layer_nr % 2) + 2) % 2 == 1) //Add extra insets every 2 layers when spiralizing. This makes bottoms of cups watertight.
{
inset_count += 5;
}
if (settings.get<bool>("alternate_extra_perimeter"))
{
inset_count += ((layer_nr % 2) + 2) % 2;
}
if (inset_count == 0)
{
part->insets.push_back(part->outline);
part->print_outline = part->outline;
return;
}
const coord_t wall_0_inset = settings.get<coord_t>("wall_0_inset");
coord_t line_width_0 = settings.get<coord_t>("wall_line_width_0");
coord_t line_width_x = settings.get<coord_t>("wall_line_width_x");
if (layer_nr == 0)
{
const ExtruderTrain& train_wall_0 = settings.get<ExtruderTrain&>("wall_0_extruder_nr");
line_width_0 *= train_wall_0.settings.get<Ratio>("initial_layer_line_width_factor");
const ExtruderTrain& train_wall_x = settings.get<ExtruderTrain&>("wall_x_extruder_nr");
line_width_x *= train_wall_x.settings.get<Ratio>("initial_layer_line_width_factor");
}
const bool recompute_outline_based_on_outer_wall = (settings.get<bool>("support_enable") || settings.get<bool>("support_tree_enable")) && !settings.get<bool>("fill_outline_gaps");
for(size_t i = 0; i < inset_count; i++)
{
part->insets.push_back(Polygons());
if (i == 0)
{
part->insets[0] = part->outline.offset(-line_width_0 / 2 - wall_0_inset);
}
else if (i == 1)
{
part->insets[1] = part->insets[0].offset(-line_width_0 / 2 + wall_0_inset - line_width_x / 2);
}
else
{
part->insets[i] = part->insets[i - 1].offset(-line_width_x);
}
const size_t inset_part_count = part->insets[i].size();
constexpr size_t minimum_part_saving = 3; //Only try if the part has more pieces than the previous inset and saves at least this many parts.
constexpr coord_t try_smaller = 10; //How many micrometres to inset with the try with a smaller inset.
if (inset_part_count > minimum_part_saving + 1 && (i == 0 || (i > 0 && inset_part_count > part->insets[i - 1].size() + minimum_part_saving)))
{
//Try a different line thickness and see if this fits better, based on these criteria:
// - There are fewer parts to the polygon (fits better in slim areas).
// - The polygon area is largely unaffected.
Polygons alternative_inset;
if (i == 0)
{
alternative_inset = part->outline.offset(-(line_width_0 - try_smaller) / 2 - wall_0_inset);
}
else if (i == 1)
{
alternative_inset = part->insets[0].offset(-(line_width_0 - try_smaller) / 2 + wall_0_inset - line_width_x / 2);
}
else
{
alternative_inset = part->insets[i - 1].offset(-(line_width_x - try_smaller));
}
if (alternative_inset.size() < inset_part_count - minimum_part_saving) //Significantly fewer parts (saves more than 3 parts).
{
part->insets[i] = alternative_inset;
}
}
//Finally optimize all the polygons. Every point removed saves time in the long run.
part->insets[i].simplify();
part->insets[i].removeDegenerateVerts();
if (i == 0)
{
if (recompute_outline_based_on_outer_wall)
{
part->print_outline = part->insets[0].offset(line_width_0 / 2, ClipperLib::jtSquare);
}
else
{
part->print_outline = part->outline;
}
}
if (part->insets[i].size() < 1)
{
part->insets.pop_back();
break;
}
}
}
/*
* This function is executed in a parallel region based on layer_nr.
* When modifying make sure any changes does not introduce data races.
*
* generateInsets only reads and writes data for the current layer
*/
void WallsComputation::generateInsets(SliceLayer* layer)
{
for(unsigned int partNr = 0; partNr < layer->parts.size(); partNr++)
{
generateInsets(&layer->parts[partNr]);
}
const bool remove_parts_with_no_insets = !settings.get<bool>("fill_outline_gaps");
//Remove the parts which did not generate an inset. As these parts are too small to print,
// and later code can now assume that there is always minimal 1 inset line.
for (unsigned int part_idx = 0; part_idx < layer->parts.size(); part_idx++)
{
if (layer->parts[part_idx].insets.size() == 0 && remove_parts_with_no_insets)
{
if (part_idx != layer->parts.size() - 1)
{ // move existing part into part to be deleted
layer->parts[part_idx] = std::move(layer->parts.back());
}
layer->parts.pop_back(); // always remove last element from array (is more efficient)
part_idx -= 1; // check the part we just moved here
}
}
}
}//namespace cura