93 LocalData &local_data = data_per_thread.local();
94 for (const int tri_i : tris_range) {
95 const int3 &tri = corner_tris[tri_i];
98 const Bounds<int2> cell_bounds = tri_to_cell_bounds(tri, resolution, uv_map);
99 const TriWithRange tri_with_range{tri_i, cell_bounds.min.x, cell_bounds.max.x};
102 for (int cell_y = cell_bounds.min.y; cell_y <= cell_bounds.max.y; cell_y++) {
103 LocalRowData &row = *local_data.rows.lookup_or_add_cb(
104 cell_y, [&]() { return local_data.allocator.construct<LocalRowData>(); });
105 row.tris.append(local_data.allocator, tri_with_range);
106 row.x_min = std::min<int>(row.x_min, cell_bounds.min.x);
107 row.x_max = std::max<int>(row.x_max, cell_bounds.max.x);
123 Vector<const LocalRowData *, 32> local_rows;
124 for (const int y : all_ys.slice(all_ys_range)) {
125 Row &row = lookup_grid.rows[y - y_bounds.min];
128 for (const LocalData *local_data : local_data_vec) {
129 if (const destruct_ptr<LocalRowData> *local_row = local_data->rows.lookup_ptr(y)) {
130 local_rows.append(local_row->get());
134 int x_min = INT32_MAX;
135 int x_max = INT32_MIN;
136 for (const LocalRowData *local_row : local_rows) {
137 x_min = std::min(x_min, local_row->x_min);
138 x_max = std::max(x_max, local_row->x_max);
141 const int x_num = x_max - x_min + 1;
142 row.offsets.reinitialize(x_num + 1);
145 MutableSpan<int> counts = row.offsets;
147 for (const LocalRowData *local_row : local_rows) {
148 for (const TriWithRange &tri_with_range : local_row->tris) {
149 for (int x = tri_with_range.x_min; x <= tri_with_range.x_max; x++) {
154 offset_indices::accumulate_counts_to_offsets(counts);
156 const int tri_indices_num = row.offsets.last();
157 row.tri_indices.reinitialize(tri_indices_num);
160 Array<int, 1000> current_offsets(x_num, 0);
161 for (const LocalRowData *local_row : local_rows) {
162 for (const TriWithRange &tri_with_range : local_row->tris) {
163 for (int x = tri_with_range.x_min; x <= tri_with_range.x_max; x++) {
164 const int offset_x = x - x_min;
165 row.tri_indices[row.offsets[offset_x] + current_offsets[offset_x]] =
166 tri_with_range.tri_index;
167 current_offsets[offset_x]++;
179 : uv_map_(uv_map), corner_tris_(corner_tris), lookup_grid_(std::make_unique<
LookupGrid>())
184 resolution_ = std::max<int>(3, std::sqrt(corner_tris.
size()) * 3);
194 for (
const LocalData &local_data : data_per_thread) {
195 local_data_vec.
append(&local_data);
196 for (
const int y : local_data.rows.keys()) {
202 lookup_grid_->y_min = y_bounds.
min;
204 const int rows_num = y_bounds.
max - y_bounds.
min + 1;
205 lookup_grid_->rows.reinitialize(rows_num);
207 finish_rows(all_ys, local_data_vec, y_bounds, *lookup_grid_);
246 const float edge_epsilon = 0.00001f;
250 const float area_epsilon = 0.00001f;
252 for (
const int tri_i : tri_indices) {
253 const int3 &tri = corner_tris_[tri_i];
254 const float2 &uv_0 = uv_map_[tri[0]];
255 const float2 &uv_1 = uv_map_[tri[1]];
256 const float2 &uv_2 = uv_map_[tri[2]];
264 const float x_dist = std::max(-bary_weights.x, bary_weights.x - 1.0f);
265 const float y_dist = std::max(-bary_weights.y, bary_weights.y - 1.0f);
266 const float z_dist = std::max(-bary_weights.z, bary_weights.z - 1.0f);
267 const float dist = std::max({x_dist, y_dist, z_dist});
269 if (dist <= 0.0f && best_dist <= 0.0f) {
270 const float worse_dist = std::max(dist, best_dist);
272 if (worse_dist < -edge_epsilon) {
273 const int3 &best_tri = corner_tris_[tri_i];
275 uv_map_[best_tri[0]], uv_map_[best_tri[1]], uv_map_[best_tri[2]]);
276 const float current_tri_area =
area_tri_v2(uv_0, uv_1, uv_2);
277 if (best_tri_area > area_epsilon && current_tri_area > area_epsilon) {
284 if (dist < best_dist) {
286 best_bary_weights = bary_weights;
287 best_tri_index = tri_i;
293 if (best_dist < edge_epsilon) {