Welcome to mirror list, hosted at ThFree Co, Russian Federation.

bmesh_separate.c « tools « bmesh « blender « source - git.blender.org/blender.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
blob: b3d0c3398858c2e4577c86f8b73e2549aa526997 (plain)
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
/*
 * ***** BEGIN GPL LICENSE BLOCK *****
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 *
 * ***** END GPL LICENSE BLOCK *****
 */

/** \file blender/bmesh/tools/bmesh_separate.c
 *  \ingroup bmesh
 *
 * BMesh separate, disconnects a set of faces from all others,
 * so they don't share any vertices/edges with other faces.
 */

#include <limits.h>

#include "MEM_guardedalloc.h"

#include "BLI_utildefines.h"
#include "BLI_buffer.h"

#include "bmesh.h"
#include "intern/bmesh_private.h"
#include "bmesh_separate.h"  /* own include */

/**
 * Split all faces that match `filter_fn`.
 * \note
 */
void BM_mesh_separate_faces(
        BMesh *bm,
        BMFaceFilterFunc filter_fn, void *user_data)
{
	BMFace **faces_array_all = MEM_mallocN(bm->totface * sizeof(BMFace *), __func__);
	/*
	 * - Create an array of faces based on 'filter_fn'.
	 *   First part of array for match, for non-match.
	 *
	 * - Enable all vertex tags, then clear all tagged vertices from 'faces_b'.
	 *
	 * - Loop over 'faces_a', checking each vertex,
	 *   splitting out any which aren't tagged (and therefor shared), disabling tags as we go.
	 */

	BMFace *f;
	BMIter iter;

	uint faces_a_len = 0;
	uint faces_b_len = 0;
	{
		int i_a = 0;
		int i_b = bm->totface;
		BM_ITER_MESH (f, &iter, bm, BM_FACES_OF_MESH) {
			faces_array_all[filter_fn(f, user_data) ? i_a++ : --i_b] = f;
		}
		faces_a_len = i_a;
		faces_b_len = bm->totface - i_a;
	}

	BMFace **faces_a = faces_array_all;
	BMFace **faces_b = faces_array_all + faces_a_len;

	/* Enable for all  */
	BM_mesh_elem_hflag_enable_all(bm, BM_VERT, BM_ELEM_TAG, false);

	/* Disable vert tag on faces_b */
	for (uint i = 0; i < faces_b_len; i++) {
		BMLoop *l_iter, *l_first;
		l_iter = l_first = BM_FACE_FIRST_LOOP(faces_b[i]);
		do {
			BM_elem_flag_disable(l_iter->v, BM_ELEM_TAG);
		} while ((l_iter = l_iter->next) != l_first);
	}


	BLI_buffer_declare_static(BMLoop **, loop_split, 0, 128);

	/* Check shared verts ('faces_a' tag and disable) */
	for (uint i = 0; i < faces_a_len; i++) {
		BMLoop *l_iter, *l_first;
		l_iter = l_first = BM_FACE_FIRST_LOOP(faces_a[i]);
		do {
			if (!BM_elem_flag_test(l_iter->v, BM_ELEM_TAG)) {
				BMVert *v = l_iter->v;
				/* Enable, since we may visit this vertex again on other faces */
				BM_elem_flag_enable(v, BM_ELEM_TAG);

				/* We know the vertex is shared, collect all vertices and split them off. */

				/* Fill 'loop_split' */
				{
					BMEdge *e_first, *e_iter;
					e_iter = e_first = l_iter->e;
					do {
						if (e_iter->l != NULL) {
							BMLoop *l_radial_first, *l_radial_iter;
							l_radial_first = l_radial_iter = e_iter->l;
							do {
								if (l_radial_iter->v == v) {
									if (filter_fn(l_radial_iter->f, user_data)) {
										BLI_buffer_append(&loop_split, BMLoop *, l_radial_iter);
									}
								}
							} while ((l_radial_iter = l_radial_iter->radial_next) != l_radial_first);
						}
					} while ((e_iter = bmesh_disk_edge_next(e_iter, v)) != e_first);
				}

				/* Perform the split */
				BM_face_loop_separate_multi(bm, loop_split.data, loop_split.count);

				BLI_buffer_clear(&loop_split);
			}
		} while ((l_iter = l_iter->next) != l_first);
	}

	BLI_buffer_free(&loop_split);

	MEM_freeN(faces_array_all);
}