| 
									
										
										
										
											2011-10-10 09:38:02 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2008-01-07 19:13:47 +00:00
										 |  |  |  * ***** BEGIN GPL LICENSE BLOCK ***** | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * 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 | 
					
						
							| 
									
										
										
										
											2008-01-07 19:13:47 +00:00
										 |  |  |  * of the License, or (at your option) any later version. | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * 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, | 
					
						
							| 
									
										
										
										
											2010-02-12 13:34:04 +00:00
										 |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV. | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  |  * Contributor(s): Blender Foundation | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2008-01-07 19:13:47 +00:00
										 |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-27 20:40:57 +00:00
										 |  |  | /** \file blender/blenkernel/intern/mesh.c
 | 
					
						
							|  |  |  |  *  \ingroup bke | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | #include "DNA_scene_types.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "DNA_material_types.h"
 | 
					
						
							| 
									
										
										
										
											2016-07-20 16:29:54 +02:00
										 |  |  | #include "DNA_meta_types.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "DNA_object_types.h"
 | 
					
						
							|  |  |  | #include "DNA_key_types.h"
 | 
					
						
							| 
									
										
										
										
											2012-02-19 22:17:30 +00:00
										 |  |  | #include "DNA_mesh_types.h"
 | 
					
						
							| 
									
										
										
										
											2015-11-12 18:52:58 +11:00
										 |  |  | #include "DNA_curve_types.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-09 04:06:48 +00:00
										 |  |  | #include "BLI_utildefines.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-07 18:36:47 +00:00
										 |  |  | #include "BLI_math.h"
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | #include "BLI_listbase.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-07 18:36:47 +00:00
										 |  |  | #include "BLI_edgehash.h"
 | 
					
						
							| 
									
										
										
										
											2013-09-20 11:14:08 +00:00
										 |  |  | #include "BLI_string.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-07 18:36:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-28 03:45:24 +00:00
										 |  |  | #include "BKE_animsys.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "BKE_main.h"
 | 
					
						
							| 
									
										
										
										
											2005-03-29 16:43:39 +00:00
										 |  |  | #include "BKE_DerivedMesh.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "BKE_global.h"
 | 
					
						
							|  |  |  | #include "BKE_mesh.h"
 | 
					
						
							|  |  |  | #include "BKE_displist.h"
 | 
					
						
							|  |  |  | #include "BKE_library.h"
 | 
					
						
							| 
									
										
										
										
											2016-07-08 18:08:36 +02:00
										 |  |  | #include "BKE_library_query.h"
 | 
					
						
							|  |  |  | #include "BKE_library_remap.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "BKE_material.h"
 | 
					
						
							| 
									
										
										
										
											2010-09-09 00:14:51 +00:00
										 |  |  | #include "BKE_modifier.h"
 | 
					
						
							|  |  |  | #include "BKE_multires.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | #include "BKE_key.h"
 | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | #include "BKE_mball.h"
 | 
					
						
							|  |  |  | #include "BKE_depsgraph.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | /* these 2 are only used by conversion functions */ | 
					
						
							|  |  |  | #include "BKE_curve.h"
 | 
					
						
							|  |  |  | /* -- */ | 
					
						
							|  |  |  | #include "BKE_object.h"
 | 
					
						
							| 
									
										
										
										
											2013-04-13 20:31:52 +00:00
										 |  |  | #include "BKE_editmesh.h"
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Depsgraph: New dependency graph integration commit
This commit integrates the work done so far on the new dependency graph system,
where goal was to replace legacy depsgraph with the new one, supporting loads of
neat features like:
- More granular dependency relation nature, which solves issues with fake cycles
  in the dependencies.
- Move towards all-animatable, by better integration of drivers into the system.
- Lay down some basis for upcoming copy-on-write, overrides and so on.
The new system is living side-by-side with the previous one and disabled by
default, so nothing will become suddenly broken. The way to enable new depsgraph
is to pass `--new-depsgraph` command line argument.
It's a bit early to consider the system production-ready, there are some TODOs
and issues were discovered during the merge period, they'll be addressed ASAP.
But it's important to merge, because it's the only way to attract artists to
really start testing this system.
There are number of assorted documents related on the design of the new system:
* http://wiki.blender.org/index.php/User:Aligorith/GSoC2013_Depsgraph#Design_Documents
* http://wiki.blender.org/index.php/User:Nazg-gul/DependencyGraph
There are also some user-related information online:
* http://code.blender.org/2015/02/blender-dependency-graph-branch-for-users/
* http://code.blender.org/2015/03/more-dependency-graph-tricks/
Kudos to everyone who was involved into the project:
- Joshua "Aligorith" Leung -- design specification, initial code
- Lukas "lukas_t" Toenne -- integrating code into blender, with further fixes
- Sergey "Sergey" "Sharybin" -- some mocking around, trying to wrap up the
  project and so
- Bassam "slikdigit" Kurdali -- stressing the new system, reporting all the
  issues and recording/writing documentation.
- Everyone else who i forgot to mention here :)
											
										 
											2015-05-12 15:05:57 +05:00
										 |  |  | #include "DEG_depsgraph.h"
 | 
					
						
							| 
									
										
										
										
											2009-05-26 04:17:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | /* Define for cases when you want extra validation of mesh
 | 
					
						
							|  |  |  |  * after certain modifications. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | // #undef VALIDATE_MESH
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | enum { | 
					
						
							|  |  |  | 	MESHCMP_DVERT_WEIGHTMISMATCH = 1, | 
					
						
							|  |  |  | 	MESHCMP_DVERT_GROUPMISMATCH, | 
					
						
							|  |  |  | 	MESHCMP_DVERT_TOTGROUPMISMATCH, | 
					
						
							|  |  |  | 	MESHCMP_LOOPCOLMISMATCH, | 
					
						
							|  |  |  | 	MESHCMP_LOOPUVMISMATCH, | 
					
						
							|  |  |  | 	MESHCMP_LOOPMISMATCH, | 
					
						
							|  |  |  | 	MESHCMP_POLYVERTMISMATCH, | 
					
						
							|  |  |  | 	MESHCMP_POLYMISMATCH, | 
					
						
							|  |  |  | 	MESHCMP_EDGEUNKNOWN, | 
					
						
							|  |  |  | 	MESHCMP_VERTCOMISMATCH, | 
					
						
							| 
									
										
										
										
											2012-05-21 06:33:45 +00:00
										 |  |  | 	MESHCMP_CDLAYERS_MISMATCH | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-05-09 05:09:07 +00:00
										 |  |  | static const char *cmpcode_to_str(int code) | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	switch (code) { | 
					
						
							|  |  |  | 		case MESHCMP_DVERT_WEIGHTMISMATCH: | 
					
						
							|  |  |  | 			return "Vertex Weight Mismatch"; | 
					
						
							|  |  |  | 		case MESHCMP_DVERT_GROUPMISMATCH: | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			return "Vertex Group Mismatch"; | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 		case MESHCMP_DVERT_TOTGROUPMISMATCH: | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			return "Vertex Doesn't Belong To Same Number Of Groups"; | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 		case MESHCMP_LOOPCOLMISMATCH: | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			return "Vertex Color Mismatch"; | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 		case MESHCMP_LOOPUVMISMATCH: | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			return "UV Mismatch"; | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 		case MESHCMP_LOOPMISMATCH: | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			return "Loop Mismatch"; | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 		case MESHCMP_POLYVERTMISMATCH: | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			return "Loop Vert Mismatch In Poly Test"; | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 		case MESHCMP_POLYMISMATCH: | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			return "Loop Vert Mismatch"; | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 		case MESHCMP_EDGEUNKNOWN: | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			return "Edge Mismatch"; | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 		case MESHCMP_VERTCOMISMATCH: | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			return "Vertex Coordinate Mismatch"; | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 		case MESHCMP_CDLAYERS_MISMATCH: | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			return "CustomData Layer Count Mismatch"; | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 		default: | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			return "Mesh Comparison Code Unknown"; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-03 20:19:11 +00:00
										 |  |  | /* thresh is threshold for comparing vertices, uvs, vertex colors,
 | 
					
						
							|  |  |  |  * weights, etc.*/ | 
					
						
							| 
									
										
										
										
											2013-05-08 12:55:36 +00:00
										 |  |  | static int customdata_compare(CustomData *c1, CustomData *c2, Mesh *m1, Mesh *m2, const float thresh) | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-08 12:55:36 +00:00
										 |  |  | 	const float thresh_sq = thresh * thresh; | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 	CustomDataLayer *l1, *l2; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	int i, i1 = 0, i2 = 0, tot, j; | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	for (i = 0; i < c1->totlayer; i++) { | 
					
						
							| 
									
										
										
										
											2014-07-20 01:30:29 +10:00
										 |  |  | 		if (ELEM(c1->layers[i].type, CD_MVERT, CD_MEDGE, CD_MPOLY, | 
					
						
							|  |  |  | 		         CD_MLOOPUV, CD_MLOOPCOL, CD_MTEXPOLY, CD_MDEFORMVERT)) | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 			i1++; | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < c2->totlayer; i++) { | 
					
						
							| 
									
										
										
										
											2014-07-20 01:30:29 +10:00
										 |  |  | 		if (ELEM(c2->layers[i].type, CD_MVERT, CD_MEDGE, CD_MPOLY, | 
					
						
							|  |  |  | 		         CD_MLOOPUV, CD_MLOOPCOL, CD_MTEXPOLY, CD_MDEFORMVERT)) | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 			i2++; | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 	if (i1 != i2) | 
					
						
							|  |  |  | 		return MESHCMP_CDLAYERS_MISMATCH; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	l1 = c1->layers; l2 = c2->layers; | 
					
						
							|  |  |  | 	tot = i1; | 
					
						
							| 
									
										
										
										
											2014-07-20 01:30:29 +10:00
										 |  |  | 	i1 = 0; i2 = 0; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	for (i = 0; i < tot; i++) { | 
					
						
							| 
									
										
										
										
											2014-07-20 01:30:29 +10:00
										 |  |  | 		while (i1 < c1->totlayer && !ELEM(l1->type, CD_MVERT, CD_MEDGE, CD_MPOLY, | 
					
						
							|  |  |  | 		                                  CD_MLOOPUV, CD_MLOOPCOL, CD_MTEXPOLY, CD_MDEFORMVERT)) | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2016-03-05 09:09:05 +11:00
										 |  |  | 			i1++; | 
					
						
							|  |  |  | 			l1++; | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-20 01:30:29 +10:00
										 |  |  | 		while (i2 < c2->totlayer && !ELEM(l2->type, CD_MVERT, CD_MEDGE, CD_MPOLY, | 
					
						
							|  |  |  | 		                                  CD_MLOOPUV, CD_MLOOPCOL, CD_MTEXPOLY, CD_MDEFORMVERT)) | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2016-03-05 09:09:05 +11:00
										 |  |  | 			i2++; | 
					
						
							|  |  |  | 			l2++; | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		if (l1->type == CD_MVERT) { | 
					
						
							|  |  |  | 			MVert *v1 = l1->data; | 
					
						
							|  |  |  | 			MVert *v2 = l2->data; | 
					
						
							|  |  |  | 			int vtot = m1->totvert; | 
					
						
							|  |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			for (j = 0; j < vtot; j++, v1++, v2++) { | 
					
						
							| 
									
										
										
										
											2014-02-02 01:36:40 +11:00
										 |  |  | 				if (len_squared_v3v3(v1->co, v2->co) > thresh_sq) | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 					return MESHCMP_VERTCOMISMATCH; | 
					
						
							| 
									
										
										
										
											2012-08-17 14:43:20 +00:00
										 |  |  | 				/* I don't care about normals, let's just do coodinates */ | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		/*we're order-agnostic for edges here*/ | 
					
						
							|  |  |  | 		if (l1->type == CD_MEDGE) { | 
					
						
							|  |  |  | 			MEdge *e1 = l1->data; | 
					
						
							|  |  |  | 			MEdge *e2 = l2->data; | 
					
						
							|  |  |  | 			int etot = m1->totedge; | 
					
						
							| 
									
										
										
										
											2013-08-24 13:47:57 +00:00
										 |  |  | 			EdgeHash *eh = BLI_edgehash_new_ex(__func__, etot); | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			for (j = 0; j < etot; j++, e1++) { | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 				BLI_edgehash_insert(eh, e1->v1, e1->v2, e1); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			for (j = 0; j < etot; j++, e2++) { | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 				if (!BLI_edgehash_lookup(eh, e2->v1, e2->v2)) | 
					
						
							|  |  |  | 					return MESHCMP_EDGEUNKNOWN; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			BLI_edgehash_free(eh, NULL); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		if (l1->type == CD_MPOLY) { | 
					
						
							|  |  |  | 			MPoly *p1 = l1->data; | 
					
						
							|  |  |  | 			MPoly *p2 = l2->data; | 
					
						
							|  |  |  | 			int ptot = m1->totpoly; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			for (j = 0; j < ptot; j++, p1++, p2++) { | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 				MLoop *lp1, *lp2; | 
					
						
							|  |  |  | 				int k; | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				if (p1->totloop != p2->totloop) | 
					
						
							|  |  |  | 					return MESHCMP_POLYMISMATCH; | 
					
						
							|  |  |  | 				 | 
					
						
							|  |  |  | 				lp1 = m1->mloop + p1->loopstart; | 
					
						
							|  |  |  | 				lp2 = m2->mloop + p2->loopstart; | 
					
						
							|  |  |  | 				 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				for (k = 0; k < p1->totloop; k++, lp1++, lp2++) { | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 					if (lp1->v != lp2->v) | 
					
						
							|  |  |  | 						return MESHCMP_POLYVERTMISMATCH; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (l1->type == CD_MLOOP) { | 
					
						
							|  |  |  | 			MLoop *lp1 = l1->data; | 
					
						
							|  |  |  | 			MLoop *lp2 = l2->data; | 
					
						
							|  |  |  | 			int ltot = m1->totloop; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			for (j = 0; j < ltot; j++, lp1++, lp2++) { | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 				if (lp1->v != lp2->v) | 
					
						
							|  |  |  | 					return MESHCMP_LOOPMISMATCH; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (l1->type == CD_MLOOPUV) { | 
					
						
							|  |  |  | 			MLoopUV *lp1 = l1->data; | 
					
						
							|  |  |  | 			MLoopUV *lp2 = l2->data; | 
					
						
							|  |  |  | 			int ltot = m1->totloop; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			for (j = 0; j < ltot; j++, lp1++, lp2++) { | 
					
						
							| 
									
										
										
										
											2013-05-08 12:55:36 +00:00
										 |  |  | 				if (len_squared_v2v2(lp1->uv, lp2->uv) > thresh_sq) | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 					return MESHCMP_LOOPUVMISMATCH; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		if (l1->type == CD_MLOOPCOL) { | 
					
						
							|  |  |  | 			MLoopCol *lp1 = l1->data; | 
					
						
							|  |  |  | 			MLoopCol *lp2 = l2->data; | 
					
						
							|  |  |  | 			int ltot = m1->totloop; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			for (j = 0; j < ltot; j++, lp1++, lp2++) { | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 				if (ABS(lp1->r - lp2->r) > thresh ||  | 
					
						
							|  |  |  | 				    ABS(lp1->g - lp2->g) > thresh ||  | 
					
						
							|  |  |  | 				    ABS(lp1->b - lp2->b) > thresh ||  | 
					
						
							|  |  |  | 				    ABS(lp1->a - lp2->a) > thresh) | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					return MESHCMP_LOOPCOLMISMATCH; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (l1->type == CD_MDEFORMVERT) { | 
					
						
							|  |  |  | 			MDeformVert *dv1 = l1->data; | 
					
						
							|  |  |  | 			MDeformVert *dv2 = l2->data; | 
					
						
							|  |  |  | 			int dvtot = m1->totvert; | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			for (j = 0; j < dvtot; j++, dv1++, dv2++) { | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 				int k; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				MDeformWeight *dw1 = dv1->dw, *dw2 = dv2->dw; | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 				 | 
					
						
							|  |  |  | 				if (dv1->totweight != dv2->totweight) | 
					
						
							|  |  |  | 					return MESHCMP_DVERT_TOTGROUPMISMATCH; | 
					
						
							|  |  |  | 				 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				for (k = 0; k < dv1->totweight; k++, dw1++, dw2++) { | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 					if (dw1->def_nr != dw2->def_nr) | 
					
						
							|  |  |  | 						return MESHCMP_DVERT_GROUPMISMATCH; | 
					
						
							| 
									
										
										
										
											2013-12-01 13:11:12 +11:00
										 |  |  | 					if (fabsf(dw1->weight - dw2->weight) > thresh) | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 						return MESHCMP_DVERT_WEIGHTMISMATCH; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-11-11 23:36:56 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Used for unit testing; compares two meshes, checking only | 
					
						
							|  |  |  |  * differences we care about.  should be usable with leaf's | 
					
						
							|  |  |  |  * testing framework I get RNA work done, will use hackish | 
					
						
							|  |  |  |  * testing code for now. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | const char *BKE_mesh_cmp(Mesh *me1, Mesh *me2, float thresh) | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int c; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (!me1 || !me2) | 
					
						
							|  |  |  | 		return "Requires two input meshes"; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (me1->totvert != me2->totvert)  | 
					
						
							|  |  |  | 		return "Number of verts don't match"; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (me1->totedge != me2->totedge) | 
					
						
							|  |  |  | 		return "Number of edges don't match"; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if (me1->totpoly != me2->totpoly) | 
					
						
							|  |  |  | 		return "Number of faces don't match"; | 
					
						
							|  |  |  | 				 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	if (me1->totloop != me2->totloop) | 
					
						
							| 
									
										
										
										
											2010-10-27 02:22:55 +00:00
										 |  |  | 		return "Number of loops don't match"; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	if ((c = customdata_compare(&me1->vdata, &me2->vdata, me1, me2, thresh))) | 
					
						
							|  |  |  | 		return cmpcode_to_str(c); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((c = customdata_compare(&me1->edata, &me2->edata, me1, me2, thresh))) | 
					
						
							|  |  |  | 		return cmpcode_to_str(c); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((c = customdata_compare(&me1->ldata, &me2->ldata, me1, me2, thresh))) | 
					
						
							|  |  |  | 		return cmpcode_to_str(c); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if ((c = customdata_compare(&me1->pdata, &me2->pdata, me1, me2, thresh))) | 
					
						
							|  |  |  | 		return cmpcode_to_str(c); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-02 16:05:54 +00:00
										 |  |  | static void mesh_ensure_tessellation_customdata(Mesh *me) | 
					
						
							| 
									
										
										
										
											2009-08-30 21:30:07 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-20 05:04:51 +00:00
										 |  |  | 	if (UNLIKELY((me->totface != 0) && (me->totpoly == 0))) { | 
					
						
							|  |  |  | 		/* Pass, otherwise this function  clears 'mface' before
 | 
					
						
							|  |  |  | 		 * versioning 'mface -> mpoly' code kicks in [#30583] | 
					
						
							|  |  |  | 		 * | 
					
						
							|  |  |  | 		 * Callers could also check but safer to do here - campbell */ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		const int tottex_original = CustomData_number_of_layers(&me->pdata, CD_MTEXPOLY); | 
					
						
							|  |  |  | 		const int totcol_original = CustomData_number_of_layers(&me->ldata, CD_MLOOPCOL); | 
					
						
							| 
									
										
										
										
											2009-08-30 21:30:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-20 05:04:51 +00:00
										 |  |  | 		const int tottex_tessface = CustomData_number_of_layers(&me->fdata, CD_MTFACE); | 
					
						
							|  |  |  | 		const int totcol_tessface = CustomData_number_of_layers(&me->fdata, CD_MCOL); | 
					
						
							| 
									
										
										
										
											2011-12-06 09:28:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-20 05:04:51 +00:00
										 |  |  | 		if (tottex_tessface != tottex_original || | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		    totcol_tessface != totcol_original) | 
					
						
							| 
									
										
										
										
											2012-03-20 05:04:51 +00:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 			BKE_mesh_tessface_clear(me); | 
					
						
							| 
									
										
										
										
											2009-08-30 21:30:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-20 05:04:51 +00:00
										 |  |  | 			CustomData_from_bmeshpoly(&me->fdata, &me->pdata, &me->ldata, me->totface); | 
					
						
							| 
									
										
										
										
											2011-12-06 09:28:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-12 14:36:57 +00:00
										 |  |  | 			/* TODO - add some --debug-mesh option */ | 
					
						
							|  |  |  | 			if (G.debug & G_DEBUG) { | 
					
						
							| 
									
										
										
										
											2012-04-25 06:06:40 +00:00
										 |  |  | 				/* note: this warning may be un-called for if we are initializing the mesh for the
 | 
					
						
							| 
									
										
										
										
											2012-04-12 14:36:57 +00:00
										 |  |  | 				 * first time from bmesh, rather then giving a warning about this we could be smarter | 
					
						
							|  |  |  | 				 * and check if there was any data to begin with, for now just print the warning with | 
					
						
							|  |  |  | 				 * some info to help troubleshoot whats going on - campbell */ | 
					
						
							|  |  |  | 				printf("%s: warning! Tessellation uvs or vcol data got out of sync, " | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				       "had to reset!\n    CD_MTFACE: %d != CD_MTEXPOLY: %d || CD_MCOL: %d != CD_MLOOPCOL: %d\n", | 
					
						
							|  |  |  | 				       __func__, tottex_tessface, tottex_original, totcol_tessface, totcol_original); | 
					
						
							| 
									
										
										
										
											2012-04-12 14:36:57 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2012-03-20 05:04:51 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2009-08-30 21:30:07 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-16 21:55:30 +11:00
										 |  |  | void BKE_mesh_ensure_skin_customdata(Mesh *me) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	BMesh *bm = me->edit_btmesh ? me->edit_btmesh->bm : NULL; | 
					
						
							|  |  |  | 	MVertSkin *vs; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (bm) { | 
					
						
							|  |  |  | 		if (!CustomData_has_layer(&bm->vdata, CD_MVERT_SKIN)) { | 
					
						
							|  |  |  | 			BMVert *v; | 
					
						
							|  |  |  | 			BMIter iter; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			BM_data_layer_add(bm, &bm->vdata, CD_MVERT_SKIN); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* Mark an arbitrary vertex as root */ | 
					
						
							|  |  |  | 			BM_ITER_MESH (v, &iter, bm, BM_VERTS_OF_MESH) { | 
					
						
							|  |  |  | 				vs = CustomData_bmesh_get(&bm->vdata, v->head.data, | 
					
						
							|  |  |  | 				                          CD_MVERT_SKIN); | 
					
						
							|  |  |  | 				vs->flag |= MVERT_SKIN_ROOT; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		if (!CustomData_has_layer(&me->vdata, CD_MVERT_SKIN)) { | 
					
						
							|  |  |  | 			vs = CustomData_add_layer(&me->vdata, | 
					
						
							|  |  |  | 			                          CD_MVERT_SKIN, | 
					
						
							|  |  |  | 			                          CD_DEFAULT, | 
					
						
							|  |  |  | 			                          NULL, | 
					
						
							|  |  |  | 			                          me->totvert); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* Mark an arbitrary vertex as root */ | 
					
						
							|  |  |  | 			if (vs) { | 
					
						
							|  |  |  | 				vs->flag |= MVERT_SKIN_ROOT; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-06 09:28:25 +00:00
										 |  |  | /* this ensures grouped customdata (e.g. mtexpoly and mloopuv and mtface, or
 | 
					
						
							|  |  |  |  * mloopcol and mcol) have the same relative active/render/clone/mask indices. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * note that for undo mesh data we want to skip 'ensure_tess_cd' call since | 
					
						
							| 
									
										
										
										
											2012-03-18 07:38:51 +00:00
										 |  |  |  * we don't want to store memory for tessface when its only used for older | 
					
						
							| 
									
										
										
										
											2011-12-06 09:28:25 +00:00
										 |  |  |  * versions of the mesh. - campbell*/ | 
					
						
							| 
									
										
										
										
											2013-03-17 19:55:10 +00:00
										 |  |  | static void mesh_update_linked_customdata(Mesh *me, const bool do_ensure_tess_cd) | 
					
						
							| 
									
										
										
										
											2009-08-30 21:30:07 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (me->edit_btmesh) | 
					
						
							| 
									
										
										
										
											2013-04-16 05:59:48 +00:00
										 |  |  | 		BKE_editmesh_update_linked_customdata(me->edit_btmesh); | 
					
						
							| 
									
										
										
										
											2009-08-30 21:30:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-06 09:28:25 +00:00
										 |  |  | 	if (do_ensure_tess_cd) { | 
					
						
							| 
									
										
										
										
											2012-03-02 16:05:54 +00:00
										 |  |  | 		mesh_ensure_tessellation_customdata(me); | 
					
						
							| 
									
										
										
										
											2011-12-06 09:28:25 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-08-30 21:30:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-01 09:49:27 +00:00
										 |  |  | 	CustomData_bmesh_update_active_layers(&me->fdata, &me->pdata, &me->ldata); | 
					
						
							| 
									
										
										
										
											2009-08-30 21:30:07 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2009-05-23 03:24:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-17 19:55:10 +00:00
										 |  |  | void BKE_mesh_update_customdata_pointers(Mesh *me, const bool do_ensure_tess_cd) | 
					
						
							| 
									
										
											  
											
												
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
											
										 
											2006-11-20 04:28:02 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-12-06 09:28:25 +00:00
										 |  |  | 	mesh_update_linked_customdata(me, do_ensure_tess_cd); | 
					
						
							| 
									
										
										
										
											2009-08-30 21:30:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
											
										 
											2006-11-20 04:28:02 +00:00
										 |  |  | 	me->mvert = CustomData_get_layer(&me->vdata, CD_MVERT); | 
					
						
							|  |  |  | 	me->dvert = CustomData_get_layer(&me->vdata, CD_MDEFORMVERT); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	me->medge = CustomData_get_layer(&me->edata, CD_MEDGE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	me->mface = CustomData_get_layer(&me->fdata, CD_MFACE); | 
					
						
							|  |  |  | 	me->mcol = CustomData_get_layer(&me->fdata, CD_MCOL); | 
					
						
							|  |  |  | 	me->mtface = CustomData_get_layer(&me->fdata, CD_MTFACE); | 
					
						
							| 
									
										
										
										
											2009-07-16 06:27:37 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2009-05-26 04:17:47 +00:00
										 |  |  | 	me->mpoly = CustomData_get_layer(&me->pdata, CD_MPOLY); | 
					
						
							|  |  |  | 	me->mloop = CustomData_get_layer(&me->ldata, CD_MLOOP); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	me->mtpoly = CustomData_get_layer(&me->pdata, CD_MTEXPOLY); | 
					
						
							|  |  |  | 	me->mloopcol = CustomData_get_layer(&me->ldata, CD_MLOOPCOL); | 
					
						
							|  |  |  | 	me->mloopuv = CustomData_get_layer(&me->ldata, CD_MLOOPUV); | 
					
						
							| 
									
										
											  
											
												
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
											
										 
											2006-11-20 04:28:02 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
											
												Add Custom Loop Normals.
This is the core code for it, tools (datatransfer and modifier) will come in next commits).
RNA api is already there, though.
See the code for details, but basically, we define, for each 'smooth fan'
(which is a set of adjacent loops around a same vertex that are smooth, i.e. have a single same normal),
a 'loop normal space' (or lnor space), using auto-computed normal and relevant edges, and store
custom normal as two angular factors inside that space. This allows to have custom normals
'following' deformations of the geometry, and to only save two shorts per loop in new clnor CDLayer.
Normal manipulation (editing, mixing, interpolating, etc.) shall always happen with plain 3D vectors normals,
and be converted back into storage format at the end.
Clnor computation has also been threaded (at least for Mesh case, not for BMesh), since the process can
be rather heavy with high poly meshes.
Also, bumping subversion, and fix mess in 2.70 versioning code.
											
										 
											2015-02-05 14:24:48 +01:00
										 |  |  | bool BKE_mesh_has_custom_loop_normals(Mesh *me) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (me->edit_btmesh) { | 
					
						
							|  |  |  | 		return CustomData_has_layer(&me->edit_btmesh->bm->ldata, CD_CUSTOMLOOPNORMAL); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		return CustomData_has_layer(&me->ldata, CD_CUSTOMLOOPNORMAL); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												ID-Remap - Step one: core work (cleanup and rework of generic ID datablock handling).
This commit changes a lot of how IDs are handled internally, especially the unlinking/freeing
processes. So far, this was very fuzy, to summarize cleanly deleting or replacing a datablock
was pretty much impossible, except for a few special cases.
Also, unlinking was handled by each datatype, in a rather messy and prone-to-errors way (quite
a few ID usages were missed or wrongly handled that way).
One of the main goal of id-remap branch was to cleanup this, and fatorize ID links handling
by using library_query utils to allow generic handling of those, which is now the case
(now, generic ID links handling is only "knwon" from readfile.c and library_query.c).
This commit also adds backends to allow live replacement and deletion of datablocks in Blender
(so-called 'remapping' process, where we replace all usages of a given ID pointer by a new one,
or NULL one in case of unlinking).
This will allow nice new features, like ability to easily reload or relocate libraries, real immediate
deletion of datablocks in blender, replacement of one datablock by another, etc.
Some of those are for next commits.
A word of warning: this commit is highly risky, because it affects potentially a lot in Blender core.
Though it was tested rather deeply, being totally impossible to check all possible ID usage cases,
it's likely there are some remaining issues and bugs in new code... Please report them! ;)
Review task: D2027 (https://developer.blender.org/D2027).
Reviewed by campbellbarton, thanks a bunch.
											
										 
											2016-06-22 17:29:38 +02:00
										 |  |  | /** Free (or release) any data used by this mesh (does not free the mesh itself). */ | 
					
						
							|  |  |  | void BKE_mesh_free(Mesh *me) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
											  
											
												ID-Remap - Step one: core work (cleanup and rework of generic ID datablock handling).
This commit changes a lot of how IDs are handled internally, especially the unlinking/freeing
processes. So far, this was very fuzy, to summarize cleanly deleting or replacing a datablock
was pretty much impossible, except for a few special cases.
Also, unlinking was handled by each datatype, in a rather messy and prone-to-errors way (quite
a few ID usages were missed or wrongly handled that way).
One of the main goal of id-remap branch was to cleanup this, and fatorize ID links handling
by using library_query utils to allow generic handling of those, which is now the case
(now, generic ID links handling is only "knwon" from readfile.c and library_query.c).
This commit also adds backends to allow live replacement and deletion of datablocks in Blender
(so-called 'remapping' process, where we replace all usages of a given ID pointer by a new one,
or NULL one in case of unlinking).
This will allow nice new features, like ability to easily reload or relocate libraries, real immediate
deletion of datablocks in blender, replacement of one datablock by another, etc.
Some of those are for next commits.
A word of warning: this commit is highly risky, because it affects potentially a lot in Blender core.
Though it was tested rather deeply, being totally impossible to check all possible ID usage cases,
it's likely there are some remaining issues and bugs in new code... Please report them! ;)
Review task: D2027 (https://developer.blender.org/D2027).
Reviewed by campbellbarton, thanks a bunch.
											
										 
											2016-06-22 17:29:38 +02:00
										 |  |  | 	BKE_animdata_free(&me->id, false); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
											
										 
											2006-11-20 04:28:02 +00:00
										 |  |  | 	CustomData_free(&me->vdata, me->totvert); | 
					
						
							|  |  |  | 	CustomData_free(&me->edata, me->totedge); | 
					
						
							|  |  |  | 	CustomData_free(&me->fdata, me->totface); | 
					
						
							| 
									
										
										
										
											2009-05-26 04:17:47 +00:00
										 |  |  | 	CustomData_free(&me->ldata, me->totloop); | 
					
						
							|  |  |  | 	CustomData_free(&me->pdata, me->totpoly); | 
					
						
							| 
									
										
										
										
											2004-07-08 20:38:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												ID-Remap - Step one: core work (cleanup and rework of generic ID datablock handling).
This commit changes a lot of how IDs are handled internally, especially the unlinking/freeing
processes. So far, this was very fuzy, to summarize cleanly deleting or replacing a datablock
was pretty much impossible, except for a few special cases.
Also, unlinking was handled by each datatype, in a rather messy and prone-to-errors way (quite
a few ID usages were missed or wrongly handled that way).
One of the main goal of id-remap branch was to cleanup this, and fatorize ID links handling
by using library_query utils to allow generic handling of those, which is now the case
(now, generic ID links handling is only "knwon" from readfile.c and library_query.c).
This commit also adds backends to allow live replacement and deletion of datablocks in Blender
(so-called 'remapping' process, where we replace all usages of a given ID pointer by a new one,
or NULL one in case of unlinking).
This will allow nice new features, like ability to easily reload or relocate libraries, real immediate
deletion of datablocks in blender, replacement of one datablock by another, etc.
Some of those are for next commits.
A word of warning: this commit is highly risky, because it affects potentially a lot in Blender core.
Though it was tested rather deeply, being totally impossible to check all possible ID usage cases,
it's likely there are some remaining issues and bugs in new code... Please report them! ;)
Review task: D2027 (https://developer.blender.org/D2027).
Reviewed by campbellbarton, thanks a bunch.
											
										 
											2016-06-22 17:29:38 +02:00
										 |  |  | 	MEM_SAFE_FREE(me->mat); | 
					
						
							|  |  |  | 	MEM_SAFE_FREE(me->bb); | 
					
						
							|  |  |  | 	MEM_SAFE_FREE(me->mselect); | 
					
						
							|  |  |  | 	MEM_SAFE_FREE(me->edit_btmesh); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-06 16:22:41 +00:00
										 |  |  | static void mesh_tessface_clear_intern(Mesh *mesh, int free_customdata) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-10-31 09:50:24 +00:00
										 |  |  | 	if (free_customdata) { | 
					
						
							| 
									
										
										
										
											2012-03-06 16:22:41 +00:00
										 |  |  | 		CustomData_free(&mesh->fdata, mesh->totface); | 
					
						
							| 
									
										
										
										
											2012-10-31 09:50:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		CustomData_reset(&mesh->fdata); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-03-06 16:22:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	mesh->mface = NULL; | 
					
						
							|  |  |  | 	mesh->mtface = NULL; | 
					
						
							|  |  |  | 	mesh->mcol = NULL; | 
					
						
							|  |  |  | 	mesh->totface = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												First step to handle missing libs/datablocks when reading a file.
Idea is, instead of ignoring completely missing linked datablocks, to
create void placeholders for them.
That way, you can work on your file, save it, and find again your missing data once
lib becomes available again. Or you can edit missing lib's path (in Outliner),
save and reload the file, and you are done.
Also, Outliner now shows broken libraries (and placeholders) with a 'broken lib' icon.
Future plans are also to be able to relocate missing libs and reload them at runtime.
Code notes:
    - Placeholder ID is just a regular datablock of same type as expected linked one,
      with 'default' data, and a LIB_MISSING bitflag set.
    - To allow creation of such datablocks, creation of datablocks in BKE was split in two step:
        + Allocation of memory itself.
        + Setting of all internal data to default values.
See also the design task (T43351).
Reviewed by @campbellbarton, thanks a bunch!
Differential Revision: https://developer.blender.org/D1394
											
										 
											2015-10-20 14:44:57 +02:00
										 |  |  | void BKE_mesh_init(Mesh *me) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
											  
											
												First step to handle missing libs/datablocks when reading a file.
Idea is, instead of ignoring completely missing linked datablocks, to
create void placeholders for them.
That way, you can work on your file, save it, and find again your missing data once
lib becomes available again. Or you can edit missing lib's path (in Outliner),
save and reload the file, and you are done.
Also, Outliner now shows broken libraries (and placeholders) with a 'broken lib' icon.
Future plans are also to be able to relocate missing libs and reload them at runtime.
Code notes:
    - Placeholder ID is just a regular datablock of same type as expected linked one,
      with 'default' data, and a LIB_MISSING bitflag set.
    - To allow creation of such datablocks, creation of datablocks in BKE was split in two step:
        + Allocation of memory itself.
        + Setting of all internal data to default values.
See also the design task (T43351).
Reviewed by @campbellbarton, thanks a bunch!
Differential Revision: https://developer.blender.org/D1394
											
										 
											2015-10-20 14:44:57 +02:00
										 |  |  | 	BLI_assert(MEMCMP_STRUCT_OFS_IS_ZERO(me, id)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	me->size[0] = me->size[1] = me->size[2] = 1.0; | 
					
						
							| 
									
										
										
										
											2016-03-01 12:58:33 +01:00
										 |  |  | 	me->smoothresh = DEG2RADF(30); | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	me->texflag = ME_AUTOSPACE; | 
					
						
							| 
									
										
										
										
											2014-02-12 02:45:52 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* disable because its slow on many GPU's, see [#37518] */ | 
					
						
							|  |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	me->flag = ME_TWOSIDED; | 
					
						
							| 
									
										
										
										
											2014-02-12 02:45:52 +11:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	me->drawflag = ME_DRAWEDGES | ME_DRAWFACES | ME_DRAWCREASES; | 
					
						
							| 
									
										
										
										
											2012-10-31 15:14:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	CustomData_reset(&me->vdata); | 
					
						
							|  |  |  | 	CustomData_reset(&me->edata); | 
					
						
							|  |  |  | 	CustomData_reset(&me->fdata); | 
					
						
							|  |  |  | 	CustomData_reset(&me->pdata); | 
					
						
							|  |  |  | 	CustomData_reset(&me->ldata); | 
					
						
							| 
									
										
											  
											
												First step to handle missing libs/datablocks when reading a file.
Idea is, instead of ignoring completely missing linked datablocks, to
create void placeholders for them.
That way, you can work on your file, save it, and find again your missing data once
lib becomes available again. Or you can edit missing lib's path (in Outliner),
save and reload the file, and you are done.
Also, Outliner now shows broken libraries (and placeholders) with a 'broken lib' icon.
Future plans are also to be able to relocate missing libs and reload them at runtime.
Code notes:
    - Placeholder ID is just a regular datablock of same type as expected linked one,
      with 'default' data, and a LIB_MISSING bitflag set.
    - To allow creation of such datablocks, creation of datablocks in BKE was split in two step:
        + Allocation of memory itself.
        + Setting of all internal data to default values.
See also the design task (T43351).
Reviewed by @campbellbarton, thanks a bunch!
Differential Revision: https://developer.blender.org/D1394
											
										 
											2015-10-20 14:44:57 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Mesh *BKE_mesh_add(Main *bmain, const char *name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Mesh *me; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	me = BKE_libblock_alloc(bmain, ID_ME, name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BKE_mesh_init(me); | 
					
						
							| 
									
										
										
										
											2012-10-31 15:14:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	return me; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-10 14:52:00 +02:00
										 |  |  | Mesh *BKE_mesh_copy(Main *bmain, Mesh *me) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	Mesh *men; | 
					
						
							| 
									
										
										
										
											2016-07-11 21:27:15 +02:00
										 |  |  | 	int a; | 
					
						
							| 
									
										
										
										
											2012-03-05 18:05:06 +00:00
										 |  |  | 	const int do_tessface = ((me->totface != 0) && (me->totpoly == 0)); /* only do tessface if we have no polys */ | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2016-07-10 14:52:00 +02:00
										 |  |  | 	men = BKE_libblock_copy(bmain, &me->id); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	men->mat = MEM_dupallocN(me->mat); | 
					
						
							| 
									
										
										
										
											2016-07-11 21:27:15 +02:00
										 |  |  | 	for (a = 0; a < men->totcol; a++) { | 
					
						
							|  |  |  | 		id_us_plus((ID *)men->mat[a]); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	id_us_plus((ID *)men->texcomesh); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
											
										 
											2006-11-20 04:28:02 +00:00
										 |  |  | 	CustomData_copy(&me->vdata, &men->vdata, CD_MASK_MESH, CD_DUPLICATE, men->totvert); | 
					
						
							|  |  |  | 	CustomData_copy(&me->edata, &men->edata, CD_MASK_MESH, CD_DUPLICATE, men->totedge); | 
					
						
							| 
									
										
										
										
											2009-05-26 04:17:47 +00:00
										 |  |  | 	CustomData_copy(&me->ldata, &men->ldata, CD_MASK_MESH, CD_DUPLICATE, men->totloop); | 
					
						
							|  |  |  | 	CustomData_copy(&me->pdata, &men->pdata, CD_MASK_MESH, CD_DUPLICATE, men->totpoly); | 
					
						
							| 
									
										
										
										
											2012-03-05 18:05:06 +00:00
										 |  |  | 	if (do_tessface) { | 
					
						
							|  |  |  | 		CustomData_copy(&me->fdata, &men->fdata, CD_MASK_MESH, CD_DUPLICATE, men->totface); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 		mesh_tessface_clear_intern(men, false); | 
					
						
							| 
									
										
										
										
											2012-03-05 18:05:06 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-17 19:55:10 +00:00
										 |  |  | 	BKE_mesh_update_customdata_pointers(men, do_tessface); | 
					
						
							| 
									
										
											  
											
												
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
											
										 
											2006-11-20 04:28:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	men->edit_btmesh = NULL; | 
					
						
							| 
									
										
										
										
											2006-05-10 17:22:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-30 22:04:03 +00:00
										 |  |  | 	men->mselect = MEM_dupallocN(men->mselect); | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	men->bb = MEM_dupallocN(men->bb); | 
					
						
							| 
									
										
										
										
											2016-07-11 19:13:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-09 14:44:48 +02:00
										 |  |  | 	if (me->key) { | 
					
						
							| 
									
										
										
										
											2016-07-10 14:52:00 +02:00
										 |  |  | 		men->key = BKE_key_copy(bmain, me->key); | 
					
						
							| 
									
										
										
										
											2016-07-09 14:44:48 +02:00
										 |  |  | 		men->key->from = (ID *)men; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2006-08-20 15:22:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-25 16:15:37 +02:00
										 |  |  | 	BKE_id_copy_ensure_local(bmain, &me->id, &men->id); | 
					
						
							| 
									
										
										
										
											2015-01-09 09:52:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	return men; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-01 19:07:11 +10:00
										 |  |  | BMesh *BKE_mesh_to_bmesh( | 
					
						
							|  |  |  |         Mesh *me, Object *ob, | 
					
						
							|  |  |  |         const bool add_key_index, const struct BMeshCreateParams *params) | 
					
						
							| 
									
										
										
										
											2009-05-26 04:17:47 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BMesh *bm; | 
					
						
							| 
									
										
										
										
											2013-09-24 03:31:00 +00:00
										 |  |  | 	const BMAllocTemplate allocsize = BMALLOC_TEMPLATE_FROM_ME(me); | 
					
						
							| 
									
										
										
										
											2009-05-26 04:17:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-01 19:07:11 +10:00
										 |  |  | 	bm = BM_mesh_create(&allocsize, params); | 
					
						
							| 
									
										
										
										
											2009-05-26 04:17:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-29 22:33:06 +10:00
										 |  |  | 	BM_mesh_bm_from_me( | 
					
						
							|  |  |  | 	        bm, me, (&(struct BMeshFromMeshParams){ | 
					
						
							|  |  |  | 	            .add_key_index = add_key_index, .use_shapekey = true, .active_shapekey = ob->shapenr, | 
					
						
							|  |  |  | 	        })); | 
					
						
							| 
									
										
										
										
											2009-05-26 04:17:47 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return bm; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-20 19:49:45 +02:00
										 |  |  | void BKE_mesh_make_local(Main *bmain, Mesh *me, const bool lib_local) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-07-20 19:49:45 +02:00
										 |  |  | 	BKE_id_make_local_generic(bmain, &me->id, true, lib_local); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-20 11:14:08 +00:00
										 |  |  | bool BKE_mesh_uv_cdlayer_rename_index(Mesh *me, const int poly_index, const int loop_index, const int face_index, | 
					
						
							|  |  |  |                                       const char *new_name, const bool do_tessface) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-09-29 06:47:07 +00:00
										 |  |  | 	CustomData *pdata, *ldata, *fdata; | 
					
						
							|  |  |  | 	CustomDataLayer *cdlp, *cdlu, *cdlf; | 
					
						
							| 
									
										
										
										
											2013-09-20 11:14:08 +00:00
										 |  |  | 	const int step = do_tessface ? 3 : 2; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-29 06:47:07 +00:00
										 |  |  | 	if (me->edit_btmesh) { | 
					
						
							|  |  |  | 		pdata = &me->edit_btmesh->bm->pdata; | 
					
						
							|  |  |  | 		ldata = &me->edit_btmesh->bm->ldata; | 
					
						
							|  |  |  | 		fdata = NULL;  /* No tessellated data in BMesh! */ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		pdata = &me->pdata; | 
					
						
							|  |  |  | 		ldata = &me->ldata; | 
					
						
							|  |  |  | 		fdata = &me->fdata; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	cdlp = &pdata->layers[poly_index]; | 
					
						
							|  |  |  | 	cdlu = &ldata->layers[loop_index]; | 
					
						
							| 
									
										
										
										
											2013-10-01 12:48:41 +00:00
										 |  |  | 	cdlf = fdata && do_tessface ? &fdata->layers[face_index] : NULL; | 
					
						
							| 
									
										
										
										
											2013-09-29 06:47:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-13 16:33:18 +06:00
										 |  |  | 	if (cdlp->name != new_name) { | 
					
						
							| 
									
										
										
										
											2014-01-13 16:26:10 +06:00
										 |  |  | 		/* Mesh validate passes a name from the CD layer as the new name,
 | 
					
						
							|  |  |  | 		 * Avoid memcpy from self to self in this case. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		BLI_strncpy(cdlp->name, new_name, sizeof(cdlp->name)); | 
					
						
							|  |  |  | 		CustomData_set_layer_unique_name(pdata, cdlp - pdata->layers); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-09-20 11:14:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Loop until we do have exactly the same name for all layers! */ | 
					
						
							| 
									
										
										
										
											2015-01-26 16:03:11 +01:00
										 |  |  | 	for (i = 1; !STREQ(cdlp->name, cdlu->name) || (cdlf && !STREQ(cdlp->name, cdlf->name)); i++) { | 
					
						
							| 
									
										
										
										
											2013-09-20 11:14:08 +00:00
										 |  |  | 		switch (i % step) { | 
					
						
							|  |  |  | 			case 0: | 
					
						
							|  |  |  | 				BLI_strncpy(cdlp->name, cdlu->name, sizeof(cdlp->name)); | 
					
						
							|  |  |  | 				CustomData_set_layer_unique_name(pdata, cdlp - pdata->layers); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			case 1: | 
					
						
							|  |  |  | 				BLI_strncpy(cdlu->name, cdlp->name, sizeof(cdlu->name)); | 
					
						
							|  |  |  | 				CustomData_set_layer_unique_name(ldata, cdlu - ldata->layers); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			case 2: | 
					
						
							| 
									
										
										
										
											2013-10-01 12:48:41 +00:00
										 |  |  | 				if (cdlf) { | 
					
						
							|  |  |  | 					BLI_strncpy(cdlf->name, cdlp->name, sizeof(cdlf->name)); | 
					
						
							|  |  |  | 					CustomData_set_layer_unique_name(fdata, cdlf - fdata->layers); | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2013-09-20 11:14:08 +00:00
										 |  |  | 				break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool BKE_mesh_uv_cdlayer_rename(Mesh *me, const char *old_name, const char *new_name, bool do_tessface) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-09-29 15:09:48 +00:00
										 |  |  | 	CustomData *pdata, *ldata, *fdata; | 
					
						
							|  |  |  | 	if (me->edit_btmesh) { | 
					
						
							|  |  |  | 		pdata = &me->edit_btmesh->bm->pdata; | 
					
						
							|  |  |  | 		ldata = &me->edit_btmesh->bm->ldata; | 
					
						
							|  |  |  | 		/* No tessellated data in BMesh! */ | 
					
						
							|  |  |  | 		fdata = NULL; | 
					
						
							|  |  |  | 		do_tessface = false; | 
					
						
							| 
									
										
										
										
											2013-09-20 11:14:08 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2013-09-29 15:09:48 +00:00
										 |  |  | 		pdata = &me->pdata; | 
					
						
							|  |  |  | 		ldata = &me->ldata; | 
					
						
							|  |  |  | 		fdata = &me->fdata; | 
					
						
							|  |  |  | 		do_tessface = (do_tessface && fdata->totlayer); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		const int pidx_start = CustomData_get_layer_index(pdata, CD_MTEXPOLY); | 
					
						
							|  |  |  | 		const int lidx_start = CustomData_get_layer_index(ldata, CD_MLOOPUV); | 
					
						
							|  |  |  | 		const int fidx_start = do_tessface ? CustomData_get_layer_index(fdata, CD_MTFACE) : -1; | 
					
						
							|  |  |  | 		int pidx = CustomData_get_named_layer(pdata, CD_MTEXPOLY, old_name); | 
					
						
							|  |  |  | 		int lidx = CustomData_get_named_layer(ldata, CD_MLOOPUV, old_name); | 
					
						
							|  |  |  | 		int fidx = do_tessface ? CustomData_get_named_layer(fdata, CD_MTFACE, old_name) : -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* None of those cases should happen, in theory!
 | 
					
						
							|  |  |  | 		 * Note this assume we have the same number of mtexpoly, mloopuv and mtface layers! | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		if (pidx == -1) { | 
					
						
							|  |  |  | 			if (lidx == -1) { | 
					
						
							|  |  |  | 				if (fidx == -1) { | 
					
						
							|  |  |  | 					/* No layer found with this name! */ | 
					
						
							|  |  |  | 					return false; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							| 
									
										
										
										
											2013-11-10 20:32:27 +00:00
										 |  |  | 					lidx = fidx; | 
					
						
							| 
									
										
										
										
											2013-09-29 15:09:48 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2013-11-10 20:32:27 +00:00
										 |  |  | 			pidx = lidx; | 
					
						
							| 
									
										
										
										
											2013-09-20 11:14:08 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2013-09-29 15:09:48 +00:00
										 |  |  | 		else { | 
					
						
							|  |  |  | 			if (lidx == -1) { | 
					
						
							| 
									
										
										
										
											2013-11-10 20:32:27 +00:00
										 |  |  | 				lidx = pidx; | 
					
						
							| 
									
										
										
										
											2013-09-29 15:09:48 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			if (fidx == -1 && do_tessface) { | 
					
						
							| 
									
										
										
										
											2013-11-10 20:32:27 +00:00
										 |  |  | 				fidx = pidx; | 
					
						
							| 
									
										
										
										
											2013-09-29 15:09:48 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2013-09-20 11:14:08 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2013-09-29 15:09:48 +00:00
										 |  |  | 		/* For now, we do not consider mismatch in indices (i.e. same name leading to (relative) different indices). */ | 
					
						
							| 
									
										
										
										
											2013-11-10 20:32:27 +00:00
										 |  |  | 		else if (pidx != lidx) { | 
					
						
							|  |  |  | 			lidx = pidx; | 
					
						
							| 
									
										
										
										
											2013-09-29 15:09:48 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2013-09-20 11:14:08 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-10 20:32:27 +00:00
										 |  |  | 		/* Go back to absolute indices! */ | 
					
						
							|  |  |  | 		pidx += pidx_start; | 
					
						
							|  |  |  | 		lidx += lidx_start; | 
					
						
							|  |  |  | 		if (fidx != -1) | 
					
						
							|  |  |  | 			fidx += fidx_start; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-29 15:09:48 +00:00
										 |  |  | 		return BKE_mesh_uv_cdlayer_rename_index(me, pidx, lidx, fidx, new_name, do_tessface); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-09-20 11:14:08 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | void BKE_mesh_boundbox_calc(Mesh *me, float r_loc[3], float r_size[3]) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	BoundBox *bb; | 
					
						
							|  |  |  | 	float min[3], max[3]; | 
					
						
							|  |  |  | 	float mloc[3], msize[3]; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	if (me->bb == NULL) me->bb = MEM_callocN(sizeof(BoundBox), "boundbox"); | 
					
						
							|  |  |  | 	bb = me->bb; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	if (!r_loc) r_loc = mloc; | 
					
						
							|  |  |  | 	if (!r_size) r_size = msize; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2010-08-03 00:56:43 +00:00
										 |  |  | 	INIT_MINMAX(min, max); | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | 	if (!BKE_mesh_minmax(me, min, max)) { | 
					
						
							| 
									
										
										
										
											2005-07-18 17:33:51 +00:00
										 |  |  | 		min[0] = min[1] = min[2] = -1.0f; | 
					
						
							|  |  |  | 		max[0] = max[1] = max[2] = 1.0f; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | 	mid_v3_v3v3(r_loc, min, max); | 
					
						
							| 
									
										
										
										
											2005-07-18 17:33:51 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	r_size[0] = (max[0] - min[0]) / 2.0f; | 
					
						
							|  |  |  | 	r_size[1] = (max[1] - min[1]) / 2.0f; | 
					
						
							|  |  |  | 	r_size[2] = (max[2] - min[2]) / 2.0f; | 
					
						
							| 
									
										
										
										
											2005-07-18 17:33:51 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-05-05 14:03:12 +00:00
										 |  |  | 	BKE_boundbox_init_from_minmax(bb, min, max); | 
					
						
							| 
									
										
										
										
											2013-08-19 09:58:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	bb->flag &= ~BOUNDBOX_DIRTY; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | void BKE_mesh_texspace_calc(Mesh *me) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-01 00:06:53 +00:00
										 |  |  | 	float loc[3], size[3]; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	int a; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | 	BKE_mesh_boundbox_calc(me, loc, size); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-19 22:17:30 +00:00
										 |  |  | 	if (me->texflag & ME_AUTOSPACE) { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		for (a = 0; a < 3; a++) { | 
					
						
							|  |  |  | 			if (size[a] == 0.0f) size[a] = 1.0f; | 
					
						
							|  |  |  | 			else if (size[a] > 0.0f && size[a] < 0.00001f) size[a] = 0.00001f; | 
					
						
							|  |  |  | 			else if (size[a] < 0.0f && size[a] > -0.00001f) size[a] = -0.00001f; | 
					
						
							| 
									
										
										
										
											2005-07-14 21:57:18 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-01 11:00:36 +00:00
										 |  |  | 		copy_v3_v3(me->loc, loc); | 
					
						
							|  |  |  | 		copy_v3_v3(me->size, size); | 
					
						
							|  |  |  | 		zero_v3(me->rot); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | BoundBox *BKE_mesh_boundbox_get(Object *ob) | 
					
						
							| 
									
										
										
										
											2005-07-14 18:04:27 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	Mesh *me = ob->data; | 
					
						
							| 
									
										
										
										
											2007-12-05 21:50:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 	if (ob->bb) | 
					
						
							| 
									
										
										
										
											2007-12-05 21:50:23 +00:00
										 |  |  | 		return ob->bb; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-19 09:58:28 +00:00
										 |  |  | 	if (me->bb == NULL || (me->bb->flag & BOUNDBOX_DIRTY)) { | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | 		BKE_mesh_texspace_calc(me); | 
					
						
							| 
									
										
										
										
											2013-08-19 09:58:28 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-07-14 18:04:27 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return me->bb; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | void BKE_mesh_texspace_get(Mesh *me, float r_loc[3], float r_rot[3], float r_size[3]) | 
					
						
							| 
									
										
										
										
											2005-07-14 21:57:18 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-19 09:58:28 +00:00
										 |  |  | 	if (me->bb == NULL || (me->bb->flag & BOUNDBOX_DIRTY)) { | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | 		BKE_mesh_texspace_calc(me); | 
					
						
							| 
									
										
										
										
											2005-07-14 21:57:18 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	if (r_loc) copy_v3_v3(r_loc,  me->loc); | 
					
						
							|  |  |  | 	if (r_rot) copy_v3_v3(r_rot,  me->rot); | 
					
						
							| 
									
										
										
										
											2012-02-28 14:05:00 +00:00
										 |  |  | 	if (r_size) copy_v3_v3(r_size, me->size); | 
					
						
							| 
									
										
										
										
											2005-07-14 21:57:18 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-15 11:11:36 +00:00
										 |  |  | void BKE_mesh_texspace_copy_from_object(Mesh *me, Object *ob) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	float *texloc, *texrot, *texsize; | 
					
						
							|  |  |  | 	short *texflag; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (BKE_object_obdata_texspace_get(ob, &texflag, &texloc, &texsize, &texrot)) { | 
					
						
							|  |  |  | 		me->texflag = *texflag; | 
					
						
							|  |  |  | 		copy_v3_v3(me->loc, texloc); | 
					
						
							|  |  |  | 		copy_v3_v3(me->size, texsize); | 
					
						
							|  |  |  | 		copy_v3_v3(me->rot, texrot); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-21 07:28:14 +00:00
										 |  |  | float (*BKE_mesh_orco_verts_get(Object *ob))[3] | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-07-20 04:44:02 +00:00
										 |  |  | 	Mesh *me = ob->data; | 
					
						
							| 
									
										
										
										
											2009-11-01 00:06:53 +00:00
										 |  |  | 	MVert *mvert = NULL; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	Mesh *tme = me->texcomesh ? me->texcomesh : me; | 
					
						
							| 
									
										
										
										
											2005-07-18 19:58:23 +00:00
										 |  |  | 	int a, totvert; | 
					
						
							| 
									
										
										
										
											2007-01-21 23:46:00 +00:00
										 |  |  | 	float (*vcos)[3] = NULL; | 
					
						
							| 
									
										
										
										
											2005-07-18 18:28:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-05 12:40:54 +00:00
										 |  |  | 	/* Get appropriate vertex coordinates */ | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	vcos = MEM_callocN(sizeof(*vcos) * me->totvert, "orco mesh"); | 
					
						
							| 
									
										
										
										
											2009-11-01 00:06:53 +00:00
										 |  |  | 	mvert = tme->mvert; | 
					
						
							| 
									
										
										
										
											2012-10-23 13:28:22 +00:00
										 |  |  | 	totvert = min_ii(tme->totvert, me->totvert); | 
					
						
							| 
									
										
										
										
											2009-11-01 00:06:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	for (a = 0; a < totvert; a++, mvert++) { | 
					
						
							| 
									
										
										
										
											2010-08-01 11:00:36 +00:00
										 |  |  | 		copy_v3_v3(vcos[a], mvert->co); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-07-18 18:28:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-21 07:28:14 +00:00
										 |  |  | 	return vcos; | 
					
						
							| 
									
										
										
										
											2007-12-05 12:40:54 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2005-07-18 19:58:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | void BKE_mesh_orco_verts_transform(Mesh *me, float (*orco)[3], int totvert, int invert) | 
					
						
							| 
									
										
										
										
											2007-12-05 12:40:54 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	float loc[3], size[3]; | 
					
						
							|  |  |  | 	int a; | 
					
						
							| 
									
										
										
										
											2005-07-18 19:58:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | 	BKE_mesh_texspace_get(me->texcomesh ? me->texcomesh : me, loc, NULL, size); | 
					
						
							| 
									
										
										
										
											2005-07-18 19:58:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 	if (invert) { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		for (a = 0; a < totvert; a++) { | 
					
						
							| 
									
										
										
										
											2007-12-11 20:02:21 +00:00
										 |  |  | 			float *co = orco[a]; | 
					
						
							| 
									
										
										
										
											2010-08-01 11:00:36 +00:00
										 |  |  | 			madd_v3_v3v3v3(co, loc, co, size); | 
					
						
							| 
									
										
										
										
											2007-12-11 20:02:21 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		for (a = 0; a < totvert; a++) { | 
					
						
							| 
									
										
										
										
											2007-12-11 20:02:21 +00:00
										 |  |  | 			float *co = orco[a]; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			co[0] = (co[0] - loc[0]) / size[0]; | 
					
						
							|  |  |  | 			co[1] = (co[1] - loc[1]) / size[1]; | 
					
						
							|  |  |  | 			co[2] = (co[2] - loc[2]) / size[2]; | 
					
						
							| 
									
										
										
										
											2007-12-11 20:02:21 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2005-07-18 19:58:23 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-11-08 20:14:04 +00:00
										 |  |  | /* rotates the vertices of a face in case v[2] or v[3] (vertex index) is = 0.
 | 
					
						
							| 
									
										
										
										
											2012-03-03 20:19:11 +00:00
										 |  |  |  * this is necessary to make the if (mface->v4) check for quads work */ | 
					
						
							| 
									
										
										
										
											2008-01-29 19:49:03 +00:00
										 |  |  | int test_index_face(MFace *mface, CustomData *fdata, int mfindex, int nr) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	/* first test if the face is legal */ | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	if ((mface->v3 || nr == 4) && mface->v3 == mface->v4) { | 
					
						
							|  |  |  | 		mface->v4 = 0; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		nr--; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	if ((mface->v2 || mface->v4) && mface->v2 == mface->v3) { | 
					
						
							|  |  |  | 		mface->v3 = mface->v4; | 
					
						
							|  |  |  | 		mface->v4 = 0; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		nr--; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	if (mface->v1 == mface->v2) { | 
					
						
							|  |  |  | 		mface->v2 = mface->v3; | 
					
						
							|  |  |  | 		mface->v3 = mface->v4; | 
					
						
							|  |  |  | 		mface->v4 = 0; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		nr--; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-26 20:05:38 +00:00
										 |  |  | 	/* check corrupt cases, bow-tie geometry, cant handle these because edge data wont exist so just return 0 */ | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	if (nr == 3) { | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 		if ( | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		    /* real edges */ | 
					
						
							|  |  |  | 		    mface->v1 == mface->v2 || | 
					
						
							|  |  |  | 		    mface->v2 == mface->v3 || | 
					
						
							|  |  |  | 		    mface->v3 == mface->v1) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2011-02-09 04:45:53 +00:00
										 |  |  | 			return 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	else if (nr == 4) { | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 		if ( | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		    /* real edges */ | 
					
						
							|  |  |  | 		    mface->v1 == mface->v2 || | 
					
						
							|  |  |  | 		    mface->v2 == mface->v3 || | 
					
						
							|  |  |  | 		    mface->v3 == mface->v4 || | 
					
						
							|  |  |  | 		    mface->v4 == mface->v1 || | 
					
						
							|  |  |  | 		    /* across the face */ | 
					
						
							|  |  |  | 		    mface->v1 == mface->v3 || | 
					
						
							| 
									
										
										
										
											2012-05-20 19:49:27 +00:00
										 |  |  | 		    mface->v2 == mface->v4) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2011-02-09 04:45:53 +00:00
										 |  |  | 			return 0; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 13:07:59 +00:00
										 |  |  | 	/* prevent a zero at wrong index location */ | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	if (nr == 3) { | 
					
						
							|  |  |  | 		if (mface->v3 == 0) { | 
					
						
							| 
									
										
											  
											
												
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
											
										 
											2006-11-20 04:28:02 +00:00
										 |  |  | 			static int corner_indices[4] = {1, 2, 0, 3}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-27 03:54:23 +00:00
										 |  |  | 			SWAP(unsigned int, mface->v1, mface->v2); | 
					
						
							|  |  |  | 			SWAP(unsigned int, mface->v2, mface->v3); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 			if (fdata) | 
					
						
							| 
									
										
										
										
											2016-02-28 15:16:42 +01:00
										 |  |  | 				CustomData_swap_corners(fdata, mfindex, corner_indices); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	else if (nr == 4) { | 
					
						
							|  |  |  | 		if (mface->v3 == 0 || mface->v4 == 0) { | 
					
						
							| 
									
										
											  
											
												
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
											
										 
											2006-11-20 04:28:02 +00:00
										 |  |  | 			static int corner_indices[4] = {2, 3, 0, 1}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-27 03:54:23 +00:00
										 |  |  | 			SWAP(unsigned int, mface->v1, mface->v3); | 
					
						
							|  |  |  | 			SWAP(unsigned int, mface->v2, mface->v4); | 
					
						
							| 
									
										
											  
											
												 - convert all DerivedMesh map functions to use index based
   mapping (instead of Edit{Vert,Edge,Face} pointers)
 - dropped convertToDispListMeshMapped (whew, glad of it too)
 - added DerivedMesh drawMappedFaces function
 - dropped EM suffix for DerivedMesh functions, it was neither
   particularly correct nor descriptive
 - converted test_index_mface to test_index_face that also corrects
   MCol and TFace. Good thing we had three versions of this routine,
   you never know when one might burn down.
 - removed flipnorm_mesh, not used anymore (and was incorrect to
   boot)
 - Getting face select to work with modifiers turned out to be much
   more complicated than expected. Reworked mapping architecture for
   modifiers - basically elements in a DispListMesh are now required
   to be stored in an order that corresponds exactly to original
   ordering. MVert/MEdge/MFace all have a new flag ME_XXX_STEPINDEX
   that is set on each element that is set on the first derived element
   of each original element. I can't say the code to follow these
   requirements for subsurf is particularly transparent, but on the
   upside it is a reasonably consistent and simple system that is memory
   efficient and allows keeping the DispListMesh structure.
 - rewrote mirror modifier to be simpler/conform to new requirements
   for mapped DispListMesh structure. This also means that mirror interacts
   much better with incremental subsurf calculation (it used to recalc
   one entire side on any topology change, now it generally avoids that).
 - added EM_{init,free}_index_arrays and EM_get_{vert,edge,face}_for_index
   functions to handle mapping indices back into appropriate EditMesh
   structures.
 - bug fix, make edges didn't recalc object data
 - bug fix, initial image assignment to TFace's didn't recalc object data
 - new feature, added circle select support for FACESELECT
 - bug fix, creating new faces in editmode duplicated the TFACE active
   flag - but there should only be one active tface
 - bug fix, possible crash when deleting all faces in faceselect mode
   on mesh with tfaces...
Still todo: TFace edge drawing is still not always correct in face
mode, in particular with a mirror modifier when mesh has edges (and
no preceeding subsurf). Have not yet decided how to deal with this.
Best solution is probably to do switch to meshes all having MEdge's,
in which case I can get rid of TFace edge flags (and need to recalc
modifiers on tface selection change).
											
										 
											2005-08-20 03:08:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 			if (fdata) | 
					
						
							| 
									
										
										
										
											2016-02-28 15:16:42 +01:00
										 |  |  | 				CustomData_swap_corners(fdata, mfindex, corner_indices); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-01-29 19:49:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return nr; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-05 16:03:57 +00:00
										 |  |  | Mesh *BKE_mesh_from_object(Object *ob) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	if (ob == NULL) return NULL; | 
					
						
							|  |  |  | 	if (ob->type == OB_MESH) return ob->data; | 
					
						
							| 
									
										
										
										
											2011-02-13 10:52:18 +00:00
										 |  |  | 	else return NULL; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-17 19:55:10 +00:00
										 |  |  | void BKE_mesh_assign_object(Object *ob, Mesh *me) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	Mesh *old = NULL; | 
					
						
							| 
									
										
										
										
											2010-09-09 00:14:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	multires_force_update(ob); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	if (ob == NULL) return; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	if (ob->type == OB_MESH) { | 
					
						
							|  |  |  | 		old = ob->data; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 		if (old) | 
					
						
							| 
									
										
										
										
											2015-11-09 19:47:10 +01:00
										 |  |  | 			id_us_min(&old->id); | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		ob->data = me; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		id_us_plus((ID *)me); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2016-07-08 14:36:25 +02:00
										 |  |  | 	test_object_materials(ob, (ID *)me); | 
					
						
							| 
									
										
										
										
											2010-09-09 00:14:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	test_object_modifiers(ob); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | void BKE_mesh_from_metaball(ListBase *lb, Mesh *me) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	DispList *dl; | 
					
						
							|  |  |  | 	MVert *mvert; | 
					
						
							| 
									
										
										
										
											2012-03-20 14:23:22 +00:00
										 |  |  | 	MLoop *mloop, *allloop; | 
					
						
							|  |  |  | 	MPoly *mpoly; | 
					
						
							| 
									
										
										
										
											2014-04-27 00:20:13 +10:00
										 |  |  | 	const float *nors, *verts; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	int a, *index; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	dl = lb->first; | 
					
						
							|  |  |  | 	if (dl == NULL) return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (dl->type == DL_INDEX4) { | 
					
						
							|  |  |  | 		mvert = CustomData_add_layer(&me->vdata, CD_MVERT, CD_CALLOC, NULL, dl->nr); | 
					
						
							|  |  |  | 		allloop = mloop = CustomData_add_layer(&me->ldata, CD_MLOOP, CD_CALLOC, NULL, dl->parts * 4); | 
					
						
							|  |  |  | 		mpoly = CustomData_add_layer(&me->pdata, CD_MPOLY, CD_CALLOC, NULL, dl->parts); | 
					
						
							|  |  |  | 		me->mvert = mvert; | 
					
						
							|  |  |  | 		me->mloop = mloop; | 
					
						
							|  |  |  | 		me->mpoly = mpoly; | 
					
						
							|  |  |  | 		me->totvert = dl->nr; | 
					
						
							|  |  |  | 		me->totpoly = dl->parts; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		a = dl->nr; | 
					
						
							|  |  |  | 		nors = dl->nors; | 
					
						
							|  |  |  | 		verts = dl->verts; | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 		while (a--) { | 
					
						
							| 
									
										
										
										
											2011-11-20 16:21:13 +00:00
										 |  |  | 			copy_v3_v3(mvert->co, verts); | 
					
						
							| 
									
										
										
										
											2011-03-28 02:34:55 +00:00
										 |  |  | 			normal_float_to_short_v3(mvert->no, nors); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 			mvert++; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			nors += 3; | 
					
						
							|  |  |  | 			verts += 3; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		a = dl->parts; | 
					
						
							|  |  |  | 		index = dl->index; | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 		while (a--) { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			int count = index[2] != index[3] ? 4 : 3; | 
					
						
							| 
									
										
										
										
											2006-11-23 00:26:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			mloop[0].v = index[0]; | 
					
						
							|  |  |  | 			mloop[1].v = index[1]; | 
					
						
							|  |  |  | 			mloop[2].v = index[2]; | 
					
						
							| 
									
										
										
										
											2012-03-20 14:23:22 +00:00
										 |  |  | 			if (count == 4) | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				mloop[3].v = index[3]; | 
					
						
							| 
									
										
										
										
											2006-11-23 00:26:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			mpoly->totloop = count; | 
					
						
							|  |  |  | 			mpoly->loopstart = (int)(mloop - allloop); | 
					
						
							|  |  |  | 			mpoly->flag = ME_SMOOTH; | 
					
						
							| 
									
										
										
										
											2012-03-20 14:23:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			mpoly++; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			mloop += count; | 
					
						
							|  |  |  | 			me->totloop += count; | 
					
						
							|  |  |  | 			index += 4; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
											  
											
												
Added custom vertex/edge/face data for meshes:
All data layers, including MVert/MEdge/MFace, are now managed as custom
data layers. The pointers like Mesh.mvert, Mesh.dvert or Mesh.mcol are
still used of course, but allocating, copying or freeing these arrays
should be done through the CustomData API.
Work in progress documentation on this is here:
http://mediawiki.blender.org/index.php/BlenderDev/BlenderArchitecture/CustomData
Replaced TFace by MTFace:
This is the same struct, except that it does not contain color, that now
always stays separated in MCol. This was not a good design decision to
begin with, and it is needed for adding multiple color layers later. Note
that this does mean older Blender versions will not be able to read UV
coordinates from the next release, due to an SDNA limitation.
Removed DispListMesh:
This now fully replaced by DerivedMesh. To provide access to arrays of
vertices, edges and faces, like DispListMesh does. The semantics of the
DerivedMesh.getVertArray() and similar functions were changed to return
a pointer to an array if one exists, or otherwise allocate a temporary
one. On releasing the DerivedMesh, this temporary array will be removed
automatically.
Removed ssDM and meshDM DerivedMesh backends:
The ssDM backend was for DispListMesh, so that became obsolete automatically.
The meshDM backend was replaced by the custom data backend, that now figures
out which layers need to be modified, and only duplicates those.
This changes code in many places, and overall removes 2514 lines of code.
So, there's a good chance this might break some stuff, although I've been
testing it for a few days now. The good news is, adding multiple color and
uv layers should now become easy.
											
										 
											2006-11-20 04:28:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-17 19:55:10 +00:00
										 |  |  | 		BKE_mesh_update_customdata_pointers(me, true); | 
					
						
							| 
									
										
										
										
											2012-01-24 00:52:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-28 14:23:07 +00:00
										 |  |  | 		BKE_mesh_calc_normals(me); | 
					
						
							| 
									
										
										
										
											2011-11-13 15:13:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-16 01:19:03 +00:00
										 |  |  | 		BKE_mesh_calc_edges(me, true, false); | 
					
						
							| 
									
										
										
										
											2011-04-15 01:19:13 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-22 12:00:37 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Specialized function to use when we _know_ existing edges don't overlap with poly edges. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void make_edges_mdata_extend(MEdge **r_alledge, int *r_totedge, | 
					
						
							|  |  |  |                                     const MPoly *mpoly, MLoop *mloop, | 
					
						
							|  |  |  |                                     const int totpoly) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int totedge = *r_totedge; | 
					
						
							|  |  |  | 	int totedge_new; | 
					
						
							|  |  |  | 	EdgeHash *eh; | 
					
						
							| 
									
										
										
										
											2013-08-24 14:40:15 +00:00
										 |  |  | 	unsigned int eh_reserve; | 
					
						
							| 
									
										
										
										
											2013-04-22 12:00:37 +00:00
										 |  |  | 	const MPoly *mp; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-24 14:40:15 +00:00
										 |  |  | 	eh_reserve = max_ii(totedge, BLI_EDGEHASH_SIZE_GUESS_FROM_POLYS(totpoly)); | 
					
						
							|  |  |  | 	eh = BLI_edgehash_new_ex(__func__, eh_reserve); | 
					
						
							| 
									
										
										
										
											2013-04-22 12:00:37 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0, mp = mpoly; i < totpoly; i++, mp++) { | 
					
						
							|  |  |  | 		BKE_mesh_poly_edgehash_insert(eh, mp, mloop + mp->loopstart); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	totedge_new = BLI_edgehash_size(eh); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUG
 | 
					
						
							|  |  |  | 	/* ensure that theres no overlap! */ | 
					
						
							|  |  |  | 	if (totedge_new) { | 
					
						
							|  |  |  | 		MEdge *medge = *r_alledge; | 
					
						
							|  |  |  | 		for (i = 0; i < totedge; i++, medge++) { | 
					
						
							|  |  |  | 			BLI_assert(BLI_edgehash_haskey(eh, medge->v1, medge->v2) == false); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (totedge_new) { | 
					
						
							|  |  |  | 		EdgeHashIterator *ehi; | 
					
						
							|  |  |  | 		MEdge *medge; | 
					
						
							|  |  |  | 		unsigned int e_index = totedge; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		*r_alledge = medge = (*r_alledge ? MEM_reallocN(*r_alledge, sizeof(MEdge) * (totedge + totedge_new)) : | 
					
						
							|  |  |  | 		                                   MEM_callocN(sizeof(MEdge) * totedge_new, __func__)); | 
					
						
							|  |  |  | 		medge += totedge; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		totedge += totedge_new; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* --- */ | 
					
						
							|  |  |  | 		for (ehi = BLI_edgehashIterator_new(eh); | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 		     BLI_edgehashIterator_isDone(ehi) == false; | 
					
						
							| 
									
										
										
										
											2013-04-22 12:00:37 +00:00
										 |  |  | 		     BLI_edgehashIterator_step(ehi), ++medge, e_index++) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			BLI_edgehashIterator_getKey(ehi, &medge->v1, &medge->v2); | 
					
						
							|  |  |  | 			BLI_edgehashIterator_setValue(ehi, SET_UINT_IN_POINTER(e_index)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			medge->crease = medge->bweight = 0; | 
					
						
							|  |  |  | 			medge->flag = ME_EDGEDRAW | ME_EDGERENDER; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		BLI_edgehashIterator_free(ehi); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		*r_totedge = totedge; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for (i = 0, mp = mpoly; i < totpoly; i++, mp++) { | 
					
						
							|  |  |  | 			MLoop *l = &mloop[mp->loopstart]; | 
					
						
							|  |  |  | 			MLoop *l_prev = (l + (mp->totloop - 1)); | 
					
						
							|  |  |  | 			int j; | 
					
						
							|  |  |  | 			for (j = 0; j < mp->totloop; j++, l++) { | 
					
						
							|  |  |  | 				/* lookup hashed edge index */ | 
					
						
							|  |  |  | 				l_prev->e = GET_UINT_FROM_POINTER(BLI_edgehash_lookup(eh, l_prev->v, l->v)); | 
					
						
							|  |  |  | 				l_prev = l; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BLI_edgehash_free(eh, NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | /* Initialize mverts, medges and, faces for converting nurbs to mesh and derived mesh */ | 
					
						
							|  |  |  | /* return non-zero on error */ | 
					
						
							| 
									
										
										
										
											2015-07-11 04:39:27 +10:00
										 |  |  | int BKE_mesh_nurbs_to_mdata( | 
					
						
							|  |  |  |         Object *ob, MVert **r_allvert, int *r_totvert, | 
					
						
							|  |  |  |         MEdge **r_alledge, int *r_totedge, MLoop **r_allloop, MPoly **r_allpoly, | 
					
						
							|  |  |  |         int *r_totloop, int *r_totpoly) | 
					
						
							| 
									
										
										
										
											2010-03-08 13:49:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-19 09:36:40 +00:00
										 |  |  | 	ListBase disp = {NULL, NULL}; | 
					
						
							| 
									
										
										
										
											2013-08-19 09:25:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (ob->curve_cache) { | 
					
						
							|  |  |  | 		disp = ob->curve_cache->disp; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-11 04:39:27 +10:00
										 |  |  | 	return BKE_mesh_nurbs_displist_to_mdata( | 
					
						
							|  |  |  | 	        ob, &disp, | 
					
						
							|  |  |  | 	        r_allvert, r_totvert, | 
					
						
							|  |  |  | 	        r_alledge, r_totedge, | 
					
						
							|  |  |  | 	        r_allloop, r_allpoly, NULL, | 
					
						
							|  |  |  | 	        r_totloop, r_totpoly); | 
					
						
							| 
									
										
										
										
											2010-03-08 13:49:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-03 04:58:55 +00:00
										 |  |  | /* BMESH: this doesn't calculate all edges from polygons,
 | 
					
						
							|  |  |  |  * only free standing edges are calculated */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-08 13:49:13 +00:00
										 |  |  | /* Initialize mverts, medges and, faces for converting nurbs to mesh and derived mesh */ | 
					
						
							| 
									
										
										
										
											2012-07-31 13:43:26 +00:00
										 |  |  | /* use specified dispbase */ | 
					
						
							| 
									
										
										
										
											2015-07-11 04:39:27 +10:00
										 |  |  | int BKE_mesh_nurbs_displist_to_mdata( | 
					
						
							|  |  |  |         Object *ob, const ListBase *dispbase, | 
					
						
							|  |  |  |         MVert **r_allvert, int *r_totvert, | 
					
						
							|  |  |  |         MEdge **r_alledge, int *r_totedge, | 
					
						
							|  |  |  |         MLoop **r_allloop, MPoly **r_allpoly, | 
					
						
							|  |  |  |         MLoopUV **r_alluv, | 
					
						
							|  |  |  |         int *r_totloop, int *r_totpoly) | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-08 12:53:52 +00:00
										 |  |  | 	Curve *cu = ob->data; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	DispList *dl; | 
					
						
							|  |  |  | 	MVert *mvert; | 
					
						
							| 
									
										
										
										
											2011-04-15 01:19:13 +00:00
										 |  |  | 	MPoly *mpoly; | 
					
						
							|  |  |  | 	MLoop *mloop; | 
					
						
							| 
									
										
										
										
											2013-01-10 17:37:17 +00:00
										 |  |  | 	MLoopUV *mloopuv = NULL; | 
					
						
							| 
									
										
										
										
											2012-02-03 04:58:55 +00:00
										 |  |  | 	MEdge *medge; | 
					
						
							| 
									
										
										
										
											2014-04-27 00:20:13 +10:00
										 |  |  | 	const float *data; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	int a, b, ofs, vertcount, startvert, totvert = 0, totedge = 0, totloop = 0, totvlak = 0; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	int p1, p2, p3, p4, *index; | 
					
						
							| 
									
										
										
										
											2013-12-23 12:12:40 +11:00
										 |  |  | 	const bool conv_polys = ((CU_DO_2DFILL(cu) == false) ||  /* 2d polys are filled with DL_INDEX3 displists */ | 
					
						
							| 
									
										
										
										
											2013-05-08 12:53:52 +00:00
										 |  |  | 	                         (ob->type == OB_SURF));  /* surf polys are never filled */ | 
					
						
							| 
									
										
										
										
											2010-11-06 06:22:25 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 13:07:59 +00:00
										 |  |  | 	/* count */ | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	dl = dispbase->first; | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 	while (dl) { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		if (dl->type == DL_SEGM) { | 
					
						
							|  |  |  | 			totvert += dl->parts * dl->nr; | 
					
						
							|  |  |  | 			totedge += dl->parts * (dl->nr - 1); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		else if (dl->type == DL_POLY) { | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 			if (conv_polys) { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				totvert += dl->parts * dl->nr; | 
					
						
							|  |  |  | 				totedge += dl->parts * dl->nr; | 
					
						
							| 
									
										
										
										
											2010-11-06 06:22:25 +00:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		else if (dl->type == DL_SURF) { | 
					
						
							| 
									
										
										
										
											2012-02-03 04:58:55 +00:00
										 |  |  | 			int tot; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			totvert += dl->parts * dl->nr; | 
					
						
							|  |  |  | 			tot = (dl->parts - 1 + ((dl->flag & DL_CYCL_V) == 2)) * (dl->nr - 1 + (dl->flag & DL_CYCL_U)); | 
					
						
							| 
									
										
										
										
											2012-02-03 04:58:55 +00:00
										 |  |  | 			totvlak += tot; | 
					
						
							|  |  |  | 			totloop += tot * 4; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		else if (dl->type == DL_INDEX3) { | 
					
						
							| 
									
										
										
										
											2012-02-03 04:58:55 +00:00
										 |  |  | 			int tot; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			totvert += dl->nr; | 
					
						
							| 
									
										
										
										
											2012-02-03 04:58:55 +00:00
										 |  |  | 			tot = dl->parts; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			totvlak += tot; | 
					
						
							| 
									
										
										
										
											2012-02-03 04:58:55 +00:00
										 |  |  | 			totloop += tot * 3; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		dl = dl->next; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	if (totvert == 0) { | 
					
						
							| 
									
										
										
										
											2007-05-28 16:49:48 +00:00
										 |  |  | 		/* error("can't convert"); */ | 
					
						
							|  |  |  | 		/* Make Sure you check ob->data is a curve */ | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-11 04:39:27 +10:00
										 |  |  | 	*r_allvert = mvert = MEM_callocN(sizeof(MVert) * totvert, "nurbs_init mvert"); | 
					
						
							|  |  |  | 	*r_alledge = medge = MEM_callocN(sizeof(MEdge) * totedge, "nurbs_init medge"); | 
					
						
							|  |  |  | 	*r_allloop = mloop = MEM_callocN(sizeof(MLoop) * totvlak * 4, "nurbs_init mloop"); // totloop
 | 
					
						
							|  |  |  | 	*r_allpoly = mpoly = MEM_callocN(sizeof(MPoly) * totvlak, "nurbs_init mloop"); | 
					
						
							| 
									
										
										
										
											2013-01-10 17:37:17 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-11 04:39:27 +10:00
										 |  |  | 	if (r_alluv) | 
					
						
							|  |  |  | 		*r_alluv = mloopuv = MEM_callocN(sizeof(MLoopUV) * totvlak * 4, "nurbs_init mloopuv"); | 
					
						
							| 
									
										
										
										
											2011-04-15 01:19:13 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2003-04-26 13:07:59 +00:00
										 |  |  | 	/* verts and faces */ | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	vertcount = 0; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	dl = dispbase->first; | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 	while (dl) { | 
					
						
							| 
									
										
										
										
											2015-04-20 23:37:04 +10:00
										 |  |  | 		const bool is_smooth = (dl->rt & CU_SMOOTH) != 0; | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		if (dl->type == DL_SEGM) { | 
					
						
							|  |  |  | 			startvert = vertcount; | 
					
						
							|  |  |  | 			a = dl->parts * dl->nr; | 
					
						
							|  |  |  | 			data = dl->verts; | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 			while (a--) { | 
					
						
							| 
									
										
										
										
											2011-11-20 16:21:13 +00:00
										 |  |  | 				copy_v3_v3(mvert->co, data); | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				data += 3; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				vertcount++; | 
					
						
							|  |  |  | 				mvert++; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			for (a = 0; a < dl->parts; a++) { | 
					
						
							|  |  |  | 				ofs = a * dl->nr; | 
					
						
							|  |  |  | 				for (b = 1; b < dl->nr; b++) { | 
					
						
							|  |  |  | 					medge->v1 = startvert + ofs + b - 1; | 
					
						
							|  |  |  | 					medge->v2 = startvert + ofs + b; | 
					
						
							| 
									
										
										
										
											2012-04-21 13:08:26 +00:00
										 |  |  | 					medge->flag = ME_LOOSEEDGE | ME_EDGERENDER | ME_EDGEDRAW; | 
					
						
							| 
									
										
										
										
											2012-02-03 04:58:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					medge++; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		else if (dl->type == DL_POLY) { | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 			if (conv_polys) { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				startvert = vertcount; | 
					
						
							|  |  |  | 				a = dl->parts * dl->nr; | 
					
						
							|  |  |  | 				data = dl->verts; | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 				while (a--) { | 
					
						
							| 
									
										
										
										
											2011-11-20 16:21:13 +00:00
										 |  |  | 					copy_v3_v3(mvert->co, data); | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 					data += 3; | 
					
						
							| 
									
										
										
										
											2010-11-06 06:22:25 +00:00
										 |  |  | 					vertcount++; | 
					
						
							|  |  |  | 					mvert++; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				for (a = 0; a < dl->parts; a++) { | 
					
						
							|  |  |  | 					ofs = a * dl->nr; | 
					
						
							|  |  |  | 					for (b = 0; b < dl->nr; b++) { | 
					
						
							|  |  |  | 						medge->v1 = startvert + ofs + b; | 
					
						
							|  |  |  | 						if (b == dl->nr - 1) medge->v2 = startvert + ofs; | 
					
						
							|  |  |  | 						else medge->v2 = startvert + ofs + b + 1; | 
					
						
							| 
									
										
										
										
											2012-04-21 13:08:26 +00:00
										 |  |  | 						medge->flag = ME_LOOSEEDGE | ME_EDGERENDER | ME_EDGEDRAW; | 
					
						
							| 
									
										
										
										
											2012-02-03 04:58:55 +00:00
										 |  |  | 						medge++; | 
					
						
							| 
									
										
										
										
											2010-11-06 06:22:25 +00:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		else if (dl->type == DL_INDEX3) { | 
					
						
							|  |  |  | 			startvert = vertcount; | 
					
						
							|  |  |  | 			a = dl->nr; | 
					
						
							|  |  |  | 			data = dl->verts; | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 			while (a--) { | 
					
						
							| 
									
										
										
										
											2011-11-20 16:21:13 +00:00
										 |  |  | 				copy_v3_v3(mvert->co, data); | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				data += 3; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				vertcount++; | 
					
						
							|  |  |  | 				mvert++; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			a = dl->parts; | 
					
						
							|  |  |  | 			index = dl->index; | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 			while (a--) { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				mloop[0].v = startvert + index[0]; | 
					
						
							|  |  |  | 				mloop[1].v = startvert + index[2]; | 
					
						
							|  |  |  | 				mloop[2].v = startvert + index[1]; | 
					
						
							| 
									
										
										
										
											2015-07-11 04:39:27 +10:00
										 |  |  | 				mpoly->loopstart = (int)(mloop - (*r_allloop)); | 
					
						
							| 
									
										
										
										
											2012-02-03 04:58:55 +00:00
										 |  |  | 				mpoly->totloop = 3; | 
					
						
							|  |  |  | 				mpoly->mat_nr = dl->col; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-10 17:37:17 +00:00
										 |  |  | 				if (mloopuv) { | 
					
						
							|  |  |  | 					int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					for (i = 0; i < 3; i++, mloopuv++) { | 
					
						
							| 
									
										
										
										
											2013-01-12 14:28:23 +00:00
										 |  |  | 						mloopuv->uv[0] = (mloop[i].v - startvert) / (float)(dl->nr - 1); | 
					
						
							| 
									
										
										
										
											2013-01-10 17:37:17 +00:00
										 |  |  | 						mloopuv->uv[1] = 0.0f; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-20 23:37:04 +10:00
										 |  |  | 				if (is_smooth) mpoly->flag |= ME_SMOOTH; | 
					
						
							| 
									
										
										
										
											2012-02-03 04:58:55 +00:00
										 |  |  | 				mpoly++; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				mloop += 3; | 
					
						
							|  |  |  | 				index += 3; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		else if (dl->type == DL_SURF) { | 
					
						
							|  |  |  | 			startvert = vertcount; | 
					
						
							|  |  |  | 			a = dl->parts * dl->nr; | 
					
						
							|  |  |  | 			data = dl->verts; | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 			while (a--) { | 
					
						
							| 
									
										
										
										
											2011-11-20 16:21:13 +00:00
										 |  |  | 				copy_v3_v3(mvert->co, data); | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				data += 3; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				vertcount++; | 
					
						
							|  |  |  | 				mvert++; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			for (a = 0; a < dl->parts; a++) { | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				if ( (dl->flag & DL_CYCL_V) == 0 && a == dl->parts - 1) break; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				if (dl->flag & DL_CYCL_U) {         /* p2 -> p1 -> */ | 
					
						
							|  |  |  | 					p1 = startvert + dl->nr * a;    /* p4 -> p3 -> */ | 
					
						
							|  |  |  | 					p2 = p1 + dl->nr - 1;       /* -----> next row */ | 
					
						
							|  |  |  | 					p3 = p1 + dl->nr; | 
					
						
							|  |  |  | 					p4 = p2 + dl->nr; | 
					
						
							|  |  |  | 					b = 0; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 					p2 = startvert + dl->nr * a; | 
					
						
							|  |  |  | 					p1 = p2 + 1; | 
					
						
							|  |  |  | 					p4 = p2 + dl->nr; | 
					
						
							|  |  |  | 					p3 = p1 + dl->nr; | 
					
						
							|  |  |  | 					b = 1; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				if ( (dl->flag & DL_CYCL_V) && a == dl->parts - 1) { | 
					
						
							|  |  |  | 					p3 -= dl->parts * dl->nr; | 
					
						
							|  |  |  | 					p4 -= dl->parts * dl->nr; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				for (; b < dl->nr; b++) { | 
					
						
							|  |  |  | 					mloop[0].v = p1; | 
					
						
							|  |  |  | 					mloop[1].v = p3; | 
					
						
							|  |  |  | 					mloop[2].v = p4; | 
					
						
							|  |  |  | 					mloop[3].v = p2; | 
					
						
							| 
									
										
										
										
											2015-07-11 04:39:27 +10:00
										 |  |  | 					mpoly->loopstart = (int)(mloop - (*r_allloop)); | 
					
						
							| 
									
										
										
										
											2012-02-03 04:58:55 +00:00
										 |  |  | 					mpoly->totloop = 4; | 
					
						
							|  |  |  | 					mpoly->mat_nr = dl->col; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-10 17:37:17 +00:00
										 |  |  | 					if (mloopuv) { | 
					
						
							|  |  |  | 						int orco_sizeu = dl->nr - 1; | 
					
						
							|  |  |  | 						int orco_sizev = dl->parts - 1; | 
					
						
							|  |  |  | 						int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						/* exception as handled in convertblender.c too */ | 
					
						
							|  |  |  | 						if (dl->flag & DL_CYCL_U) { | 
					
						
							|  |  |  | 							orco_sizeu++; | 
					
						
							|  |  |  | 							if (dl->flag & DL_CYCL_V) | 
					
						
							|  |  |  | 								orco_sizev++; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2013-11-26 01:17:24 +06:00
										 |  |  | 						else if (dl->flag & DL_CYCL_V) { | 
					
						
							|  |  |  | 							orco_sizev++; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2013-01-10 17:37:17 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 						for (i = 0; i < 4; i++, mloopuv++) { | 
					
						
							|  |  |  | 							/* find uv based on vertex index into grid array */ | 
					
						
							|  |  |  | 							int v = mloop[i].v - startvert; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-12 14:28:23 +00:00
										 |  |  | 							mloopuv->uv[0] = (v / dl->nr) / (float)orco_sizev; | 
					
						
							|  |  |  | 							mloopuv->uv[1] = (v % dl->nr) / (float)orco_sizeu; | 
					
						
							| 
									
										
										
										
											2013-01-10 17:37:17 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 							/* cyclic correction */ | 
					
						
							| 
									
										
										
										
											2013-11-22 20:38:48 +11:00
										 |  |  | 							if ((i == 1 || i == 2) && mloopuv->uv[0] == 0.0f) | 
					
						
							|  |  |  | 								mloopuv->uv[0] = 1.0f; | 
					
						
							| 
									
										
										
										
											2013-01-10 17:37:17 +00:00
										 |  |  | 							if ((i == 0 || i == 1) && mloopuv->uv[1] == 0.0f) | 
					
						
							|  |  |  | 								mloopuv->uv[1] = 1.0f; | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2012-07-31 13:43:26 +00:00
										 |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-20 23:37:04 +10:00
										 |  |  | 					if (is_smooth) mpoly->flag |= ME_SMOOTH; | 
					
						
							| 
									
										
										
										
											2012-02-03 04:58:55 +00:00
										 |  |  | 					mpoly++; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 					mloop += 4; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 					p4 = p3; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 					p3++; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 					p2 = p1; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 					p1++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		dl = dl->next; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-04-15 01:19:13 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2013-04-22 12:00:37 +00:00
										 |  |  | 	if (totvlak) { | 
					
						
							| 
									
										
										
										
											2015-07-11 04:39:27 +10:00
										 |  |  | 		make_edges_mdata_extend(r_alledge, &totedge, | 
					
						
							|  |  |  | 		                        *r_allpoly, *r_allloop, totvlak); | 
					
						
							| 
									
										
										
										
											2013-04-22 12:00:37 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-11 04:39:27 +10:00
										 |  |  | 	*r_totpoly = totvlak; | 
					
						
							|  |  |  | 	*r_totloop = totloop; | 
					
						
							|  |  |  | 	*r_totedge = totedge; | 
					
						
							|  |  |  | 	*r_totvert = totvert; | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-31 13:43:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | /* this may fail replacing ob->data, be sure to check ob->type */ | 
					
						
							| 
									
										
										
										
											2013-03-17 19:55:10 +00:00
										 |  |  | void BKE_mesh_from_nurbs_displist(Object *ob, ListBase *dispbase, const bool use_orco_uv) | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	Main *bmain = G.main; | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | 	Object *ob1; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	DerivedMesh *dm = ob->derivedFinal; | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | 	Mesh *me; | 
					
						
							|  |  |  | 	Curve *cu; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	MVert *allvert = NULL; | 
					
						
							|  |  |  | 	MEdge *alledge = NULL; | 
					
						
							| 
									
										
										
										
											2011-04-15 01:19:13 +00:00
										 |  |  | 	MLoop *allloop = NULL; | 
					
						
							| 
									
										
										
										
											2013-01-10 17:37:17 +00:00
										 |  |  | 	MLoopUV *alluv = NULL; | 
					
						
							| 
									
										
										
										
											2011-04-15 01:19:13 +00:00
										 |  |  | 	MPoly *allpoly = NULL; | 
					
						
							| 
									
										
										
										
											2012-02-03 04:58:55 +00:00
										 |  |  | 	int totvert, totedge, totloop, totpoly; | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	cu = ob->data; | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (dm == NULL) { | 
					
						
							| 
									
										
										
										
											2012-07-14 17:30:49 +00:00
										 |  |  | 		if (BKE_mesh_nurbs_displist_to_mdata(ob, dispbase, &allvert, &totvert, | 
					
						
							|  |  |  | 		                                     &alledge, &totedge, &allloop, | 
					
						
							| 
									
										
										
										
											2013-02-02 04:58:03 +00:00
										 |  |  | 		                                     &allpoly, (use_orco_uv) ? &alluv : NULL, | 
					
						
							| 
									
										
										
										
											2013-01-10 17:37:17 +00:00
										 |  |  | 		                                     &totloop, &totpoly) != 0) | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | 			/* Error initializing */ | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* make mesh */ | 
					
						
							| 
									
										
										
										
											2016-10-03 17:38:14 +02:00
										 |  |  | 		me = BKE_mesh_add(bmain, "Mesh"); | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		me->totvert = totvert; | 
					
						
							|  |  |  | 		me->totedge = totedge; | 
					
						
							| 
									
										
										
										
											2011-04-15 01:19:13 +00:00
										 |  |  | 		me->totloop = totloop; | 
					
						
							|  |  |  | 		me->totpoly = totpoly; | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		me->mvert = CustomData_add_layer(&me->vdata, CD_MVERT, CD_ASSIGN, allvert, me->totvert); | 
					
						
							|  |  |  | 		me->medge = CustomData_add_layer(&me->edata, CD_MEDGE, CD_ASSIGN, alledge, me->totedge); | 
					
						
							|  |  |  | 		me->mloop = CustomData_add_layer(&me->ldata, CD_MLOOP, CD_ASSIGN, allloop, me->totloop); | 
					
						
							|  |  |  | 		me->mpoly = CustomData_add_layer(&me->pdata, CD_MPOLY, CD_ASSIGN, allpoly, me->totpoly); | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-10 17:37:17 +00:00
										 |  |  | 		if (alluv) { | 
					
						
							|  |  |  | 			const char *uvname = "Orco"; | 
					
						
							|  |  |  | 			me->mtpoly = CustomData_add_layer_named(&me->pdata, CD_MTEXPOLY, CD_DEFAULT, NULL, me->totpoly, uvname); | 
					
						
							|  |  |  | 			me->mloopuv = CustomData_add_layer_named(&me->ldata, CD_MLOOPUV, CD_ASSIGN, alluv, me->totloop, uvname); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-28 14:23:07 +00:00
										 |  |  | 		BKE_mesh_calc_normals(me); | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2016-10-03 17:38:14 +02:00
										 |  |  | 		me = BKE_mesh_add(bmain, "Mesh"); | 
					
						
							| 
									
										
										
										
											2015-02-19 00:00:23 +05:00
										 |  |  | 		DM_to_mesh(dm, me, ob, CD_MASK_MESH, false); | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	me->totcol = cu->totcol; | 
					
						
							|  |  |  | 	me->mat = cu->mat; | 
					
						
							| 
									
										
										
										
											2010-03-14 12:12:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | 	BKE_mesh_texspace_calc(me); | 
					
						
							| 
									
										
										
										
											2010-03-14 12:12:21 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	cu->mat = NULL; | 
					
						
							|  |  |  | 	cu->totcol = 0; | 
					
						
							| 
									
										
											  
											
												 - added mesh_strip_loose_faces, works in conjunction with make_edges
   to get rid of faces with MFace.v3==0
 - change all Mesh's to have ->medge now. This is forced by make_edges
   on readfile, and in the various exotic important routines, and on
   conversion back in python.
 - make python NMesh structure always have medges now (needs testing)
 - with above two changes it is guarenteed that mf->v3 is never ==0
   in main blender code (i.e., all MFace's are actually triangles
   or quads) and so I went through and removed all the historic tests
   to deal with MFace.v3==0. Equals lots of deleting, I am in heaven!
 - removed MEdge edcode flag, no longer needed
 - added experimental replacement for edge flag system
Still are some inconsistencies in FACESELECT mode edge drawing to
be ironed out.
NOTE: This commit adds an experimental edge flag calc system, based
on 10-seconds-of-thought algorithm by yours truly. Would appreciate
feedback on how this system works, esp compared to old one and esp
on complex or interesting models.
To Use: New system is enabled by setting G.rt to a value between
1 and 1000 (Value of 0 uses old system). Value 1000 is reserved for
"auto" edge, which is more or less identical to old system but also
makes sure that at least 10% of edges are drawn (solves errors for
super subdivided meshes). Values between 1 and 999 act as percent
(out of 1000) of edges that should be drawn, starting with "most
interesting" edges first. Please try it and comment!
											
										 
											2005-08-21 07:19:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 	if (ob->data) { | 
					
						
							| 
									
										
										
										
											2014-01-15 16:37:03 +01:00
										 |  |  | 		BKE_libblock_free(bmain, ob->data); | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	ob->data = me; | 
					
						
							|  |  |  | 	ob->type = OB_MESH; | 
					
						
							| 
									
										
										
										
											2010-03-05 16:47:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-04-26 13:07:59 +00:00
										 |  |  | 	/* other users */ | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	ob1 = bmain->object.first; | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 	while (ob1) { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		if (ob1->data == cu) { | 
					
						
							|  |  |  | 			ob1->type = OB_MESH; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 		 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			ob1->data = ob->data; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 			id_us_plus((ID *)ob->data); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		ob1 = ob1->id.next; | 
					
						
							| 
									
										
										
										
											2002-10-12 11:37:38 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-14 17:30:49 +00:00
										 |  |  | void BKE_mesh_from_nurbs(Object *ob) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-06-20 13:27:48 +00:00
										 |  |  | 	Curve *cu = (Curve *) ob->data; | 
					
						
							|  |  |  | 	bool use_orco_uv = (cu->flag & CU_UV_ORCO) != 0; | 
					
						
							| 
									
										
										
										
											2013-08-19 09:36:40 +00:00
										 |  |  | 	ListBase disp = {NULL, NULL}; | 
					
						
							| 
									
										
										
										
											2013-08-19 09:25:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (ob->curve_cache) { | 
					
						
							|  |  |  | 		disp = ob->curve_cache->disp; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-06-20 13:27:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-19 09:25:24 +00:00
										 |  |  | 	BKE_mesh_from_nurbs_displist(ob, &disp, use_orco_uv); | 
					
						
							| 
									
										
										
										
											2012-07-14 17:30:49 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | typedef struct EdgeLink { | 
					
						
							| 
									
										
										
										
											2014-06-10 01:18:05 +10:00
										 |  |  | 	struct EdgeLink *next, *prev; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 	void *edge; | 
					
						
							|  |  |  | } EdgeLink; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct VertLink { | 
					
						
							|  |  |  | 	Link *next, *prev; | 
					
						
							| 
									
										
										
										
											2011-12-27 03:54:23 +00:00
										 |  |  | 	unsigned int index; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | } VertLink; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-27 03:54:23 +00:00
										 |  |  | static void prependPolyLineVert(ListBase *lb, unsigned int index) | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	VertLink *vl = MEM_callocN(sizeof(VertLink), "VertLink"); | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 	vl->index = index; | 
					
						
							|  |  |  | 	BLI_addhead(lb, vl); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-12-27 03:54:23 +00:00
										 |  |  | static void appendPolyLineVert(ListBase *lb, unsigned int index) | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	VertLink *vl = MEM_callocN(sizeof(VertLink), "VertLink"); | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 	vl->index = index; | 
					
						
							|  |  |  | 	BLI_addtail(lb, vl); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-15 10:48:48 +00:00
										 |  |  | void BKE_mesh_to_curve_nurblist(DerivedMesh *dm, ListBase *nurblist, const int edge_users_test) | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-03-16 14:33:32 +00:00
										 |  |  | 	MVert       *mvert = dm->getVertArray(dm); | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	MEdge *med, *medge = dm->getEdgeArray(dm); | 
					
						
							| 
									
										
										
										
											2013-03-14 18:35:21 +00:00
										 |  |  | 	MPoly *mp,  *mpoly = dm->getPolyArray(dm); | 
					
						
							|  |  |  | 	MLoop       *mloop = dm->getLoopArray(dm); | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-16 14:33:32 +00:00
										 |  |  | 	int dm_totedge = dm->getNumEdges(dm); | 
					
						
							|  |  |  | 	int dm_totpoly = dm->getNumPolys(dm); | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 	int totedges = 0; | 
					
						
							| 
									
										
										
										
											2013-03-14 18:35:21 +00:00
										 |  |  | 	int i; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* only to detect edge polylines */ | 
					
						
							| 
									
										
										
										
											2013-03-14 18:35:21 +00:00
										 |  |  | 	int *edge_users; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ListBase edges = {NULL, NULL}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-14 18:35:21 +00:00
										 |  |  | 	/* get boundary edges */ | 
					
						
							| 
									
										
										
										
											2013-03-16 14:33:32 +00:00
										 |  |  | 	edge_users = MEM_callocN(sizeof(int) * dm_totedge, __func__); | 
					
						
							|  |  |  | 	for (i = 0, mp = mpoly; i < dm_totpoly; i++, mp++) { | 
					
						
							| 
									
										
										
										
											2013-03-14 18:35:21 +00:00
										 |  |  | 		MLoop *ml = &mloop[mp->loopstart]; | 
					
						
							|  |  |  | 		int j; | 
					
						
							|  |  |  | 		for (j = 0; j < mp->totloop; j++, ml++) { | 
					
						
							|  |  |  | 			edge_users[ml->e]++; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-14 18:35:21 +00:00
										 |  |  | 	/* create edges from all faces (so as to find edges not in any faces) */ | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	med = medge; | 
					
						
							| 
									
										
										
										
											2013-03-16 14:33:32 +00:00
										 |  |  | 	for (i = 0; i < dm_totedge; i++, med++) { | 
					
						
							| 
									
										
										
										
											2013-03-15 10:48:48 +00:00
										 |  |  | 		if (edge_users[i] == edge_users_test) { | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			EdgeLink *edl = MEM_callocN(sizeof(EdgeLink), "EdgeLink"); | 
					
						
							|  |  |  | 			edl->edge = med; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			BLI_addtail(&edges, edl);   totedges++; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-03-14 18:35:21 +00:00
										 |  |  | 	MEM_freeN(edge_users); | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 	if (edges.first) { | 
					
						
							|  |  |  | 		while (edges.first) { | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 			/* each iteration find a polyline and add this as a nurbs poly spline */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			ListBase polyline = {NULL, NULL}; /* store a list of VertLink's */ | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 			bool closed = false; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			int totpoly = 0; | 
					
						
							|  |  |  | 			MEdge *med_current = ((EdgeLink *)edges.last)->edge; | 
					
						
							|  |  |  | 			unsigned int startVert = med_current->v1; | 
					
						
							|  |  |  | 			unsigned int endVert = med_current->v2; | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 			bool ok = true; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			appendPolyLineVert(&polyline, startVert);   totpoly++; | 
					
						
							|  |  |  | 			appendPolyLineVert(&polyline, endVert);     totpoly++; | 
					
						
							|  |  |  | 			BLI_freelinkN(&edges, edges.last);          totedges--; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 			while (ok) { /* while connected edges are found... */ | 
					
						
							| 
									
										
										
										
											2014-06-10 01:18:05 +10:00
										 |  |  | 				EdgeLink *edl = edges.last; | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 				ok = false; | 
					
						
							| 
									
										
										
										
											2014-06-10 01:18:05 +10:00
										 |  |  | 				while (edl) { | 
					
						
							|  |  |  | 					EdgeLink *edl_prev = edl->prev; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 					med = edl->edge; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 					if (med->v1 == endVert) { | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 						endVert = med->v2; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 						appendPolyLineVert(&polyline, med->v2); totpoly++; | 
					
						
							|  |  |  | 						BLI_freelinkN(&edges, edl);             totedges--; | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 						ok = true; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 					else if (med->v2 == endVert) { | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 						endVert = med->v1; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 						appendPolyLineVert(&polyline, endVert); totpoly++; | 
					
						
							|  |  |  | 						BLI_freelinkN(&edges, edl);             totedges--; | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 						ok = true; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 					else if (med->v1 == startVert) { | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 						startVert = med->v2; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 						prependPolyLineVert(&polyline, startVert);  totpoly++; | 
					
						
							|  |  |  | 						BLI_freelinkN(&edges, edl);                 totedges--; | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 						ok = true; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 					else if (med->v2 == startVert) { | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 						startVert = med->v1; | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 						prependPolyLineVert(&polyline, startVert);  totpoly++; | 
					
						
							|  |  |  | 						BLI_freelinkN(&edges, edl);                 totedges--; | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 						ok = true; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2014-06-10 01:18:05 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					edl = edl_prev; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* Now we have a polyline, make into a curve */ | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			if (startVert == endVert) { | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 				BLI_freelinkN(&polyline, polyline.last); | 
					
						
							|  |  |  | 				totpoly--; | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 				closed = true; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* --- nurbs --- */ | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				Nurb *nu; | 
					
						
							|  |  |  | 				BPoint *bp; | 
					
						
							|  |  |  | 				VertLink *vl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				/* create new 'nurb' within the curve */ | 
					
						
							|  |  |  | 				nu = (Nurb *)MEM_callocN(sizeof(Nurb), "MeshNurb"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				nu->pntsu = totpoly; | 
					
						
							|  |  |  | 				nu->pntsv = 1; | 
					
						
							|  |  |  | 				nu->orderu = 4; | 
					
						
							|  |  |  | 				nu->flagu = CU_NURB_ENDPOINT | (closed ? CU_NURB_CYCLIC : 0);  /* endpoint */ | 
					
						
							|  |  |  | 				nu->resolu = 12; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				nu->bp = (BPoint *)MEM_callocN(sizeof(BPoint) * totpoly, "bpoints"); | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				/* add points */ | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 				vl = polyline.first; | 
					
						
							|  |  |  | 				for (i = 0, bp = nu->bp; i < totpoly; i++, bp++, vl = (VertLink *)vl->next) { | 
					
						
							| 
									
										
										
										
											2013-03-16 14:33:32 +00:00
										 |  |  | 					copy_v3_v3(bp->vec, mvert[vl->index].co); | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 					bp->f1 = SELECT; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 					bp->radius = bp->weight = 1.0; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				BLI_freelistN(&polyline); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				/* add nurb to curve */ | 
					
						
							| 
									
										
										
										
											2013-03-14 18:44:32 +00:00
										 |  |  | 				BLI_addtail(nurblist, nu); | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			/* --- done with nurbs --- */ | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2013-03-14 18:44:32 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BKE_mesh_to_curve(Scene *scene, Object *ob) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* make new mesh data from the original copy */ | 
					
						
							|  |  |  | 	DerivedMesh *dm = mesh_get_derived_final(scene, ob, CD_MASK_MESH); | 
					
						
							|  |  |  | 	ListBase nurblist = {NULL, NULL}; | 
					
						
							|  |  |  | 	bool needsFree = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-15 10:48:48 +00:00
										 |  |  | 	BKE_mesh_to_curve_nurblist(dm, &nurblist, 0); | 
					
						
							| 
									
										
										
										
											2013-04-08 00:25:44 +00:00
										 |  |  | 	BKE_mesh_to_curve_nurblist(dm, &nurblist, 1); | 
					
						
							| 
									
										
										
										
											2013-03-14 18:44:32 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (nurblist.first) { | 
					
						
							|  |  |  | 		Curve *cu = BKE_curve_add(G.main, ob->id.name + 2, OB_CURVE); | 
					
						
							|  |  |  | 		cu->flag |= CU_3D; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		cu->nurb = nurblist; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 19:47:10 +01:00
										 |  |  | 		id_us_min(&((Mesh *)ob->data)->id); | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 		ob->data = cu; | 
					
						
							|  |  |  | 		ob->type = OB_CURVE; | 
					
						
							| 
									
										
										
										
											2010-03-07 05:04:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* curve objects can't contain DM in usual cases, we could free memory */ | 
					
						
							| 
									
										
										
										
											2013-03-14 18:35:21 +00:00
										 |  |  | 		needsFree = true; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-07 05:04:22 +00:00
										 |  |  | 	dm->needsFree = needsFree; | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | 	dm->release(dm); | 
					
						
							| 
									
										
										
										
											2010-03-07 05:04:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (needsFree) { | 
					
						
							|  |  |  | 		ob->derivedFinal = NULL; | 
					
						
							| 
									
										
										
										
											2010-06-23 09:58:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		/* curve object could have got bounding box only in special cases */ | 
					
						
							| 
									
										
										
										
											2012-02-23 02:17:50 +00:00
										 |  |  | 		if (ob->bb) { | 
					
						
							| 
									
										
										
										
											2010-06-23 09:58:02 +00:00
										 |  |  | 			MEM_freeN(ob->bb); | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 			ob->bb = NULL; | 
					
						
							| 
									
										
										
										
											2010-06-23 09:58:02 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-03-07 05:04:22 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-10-22 23:22:05 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-14 11:29:58 +00:00
										 |  |  | void BKE_mesh_material_index_remove(Mesh *me, short index) | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-08-14 11:29:58 +00:00
										 |  |  | 	MPoly *mp; | 
					
						
							|  |  |  | 	MFace *mf; | 
					
						
							| 
									
										
										
										
											2004-03-20 23:59:57 +00:00
										 |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-14 11:29:58 +00:00
										 |  |  | 	for (mp = me->mpoly, i = 0; i < me->totpoly; i++, mp++) { | 
					
						
							|  |  |  | 		if (mp->mat_nr && mp->mat_nr >= index) { | 
					
						
							| 
									
										
										
										
											2011-04-24 08:11:44 +00:00
										 |  |  | 			mp->mat_nr--; | 
					
						
							| 
									
										
										
										
											2013-08-14 11:29:58 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2011-04-24 08:06:26 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-08-14 11:29:58 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (mf = me->mface, i = 0; i < me->totface; i++, mf++) { | 
					
						
							|  |  |  | 		if (mf->mat_nr && mf->mat_nr >= index) { | 
					
						
							| 
									
										
										
										
											2004-03-20 23:59:57 +00:00
										 |  |  | 			mf->mat_nr--; | 
					
						
							| 
									
										
										
										
											2013-08-14 11:29:58 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BKE_mesh_material_index_clear(Mesh *me) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	MPoly *mp; | 
					
						
							|  |  |  | 	MFace *mf; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (mp = me->mpoly, i = 0; i < me->totpoly; i++, mp++) { | 
					
						
							|  |  |  | 		mp->mat_nr = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (mf = me->mface, i = 0; i < me->totface; i++, mf++) { | 
					
						
							|  |  |  | 		mf->mat_nr = 0; | 
					
						
							| 
									
										
										
										
											2004-03-20 23:59:57 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-28 07:24:56 +10:00
										 |  |  | void BKE_mesh_material_remap(Mesh *me, const unsigned int *remap, unsigned int remap_len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const short remap_len_short = (short)remap_len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define MAT_NR_REMAP(n) \
 | 
					
						
							|  |  |  | 	if (n < remap_len_short) { \ | 
					
						
							|  |  |  | 		BLI_assert(n >= 0 && remap[n] < remap_len_short); \ | 
					
						
							|  |  |  | 		n = remap[n]; \ | 
					
						
							|  |  |  | 	} ((void)0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (me->edit_btmesh) { | 
					
						
							|  |  |  | 		BMEditMesh *em = me->edit_btmesh; | 
					
						
							|  |  |  | 		BMIter iter; | 
					
						
							|  |  |  | 		BMFace *efa; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		BM_ITER_MESH(efa, &iter, em->bm, BM_FACES_OF_MESH) { | 
					
						
							|  |  |  | 			MAT_NR_REMAP(efa->mat_nr); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		int i; | 
					
						
							|  |  |  | 		for (i = 0; i < me->totpoly; i++) { | 
					
						
							|  |  |  | 			MAT_NR_REMAP(me->mpoly[i].mat_nr); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #undef MAT_NR_REMAP
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-05 21:28:12 +00:00
										 |  |  | void BKE_mesh_smooth_flag_set(Object *meshOb, int enableSmooth)  | 
					
						
							| 
									
										
										
										
											2009-01-04 14:14:06 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-07-14 15:48:01 +00:00
										 |  |  | 	Mesh *me = meshOb->data; | 
					
						
							| 
									
										
										
										
											2004-03-20 23:59:57 +00:00
										 |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	for (i = 0; i < me->totpoly; i++) { | 
					
						
							| 
									
										
										
										
											2015-07-12 02:53:37 +10:00
										 |  |  | 		MPoly *mp = &me->mpoly[i]; | 
					
						
							| 
									
										
										
										
											2011-04-24 08:06:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (enableSmooth) { | 
					
						
							|  |  |  | 			mp->flag |= ME_SMOOTH; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2011-04-24 08:06:26 +00:00
										 |  |  | 			mp->flag &= ~ME_SMOOTH; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-05-06 15:15:33 +00:00
										 |  |  | 	for (i = 0; i < me->totface; i++) { | 
					
						
							| 
									
										
										
										
											2015-07-12 02:53:37 +10:00
										 |  |  | 		MFace *mf = &me->mface[i]; | 
					
						
							| 
									
										
										
										
											2004-03-20 23:59:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (enableSmooth) { | 
					
						
							|  |  |  | 			mf->flag |= ME_SMOOTH; | 
					
						
							| 
									
										
										
										
											2012-03-24 06:18:31 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2004-03-20 23:59:57 +00:00
										 |  |  | 			mf->flag &= ~ME_SMOOTH; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2005-07-17 21:24:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-27 02:25:15 +00:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  |  * Return a newly MEM_malloc'd array of all the mesh vertex locations | 
					
						
							| 
									
										
										
										
											2014-03-16 03:24:05 +11:00
										 |  |  |  * \note \a r_numVerts may be NULL | 
					
						
							| 
									
										
										
										
											2013-08-27 02:25:15 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2015-03-29 04:44:05 +11:00
										 |  |  | float (*BKE_mesh_vertexCos_get(const Mesh *me, int *r_numVerts))[3] | 
					
						
							| 
									
										
										
										
											2013-08-27 02:25:15 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | 	int i, numVerts = me->totvert; | 
					
						
							|  |  |  | 	float (*cos)[3] = MEM_mallocN(sizeof(*cos) * numVerts, "vertexcos1"); | 
					
						
							| 
									
										
										
										
											2013-08-27 02:25:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | 	if (r_numVerts) *r_numVerts = numVerts; | 
					
						
							|  |  |  | 	for (i = 0; i < numVerts; i++) | 
					
						
							|  |  |  | 		copy_v3_v3(cos[i], me->mvert[i].co); | 
					
						
							| 
									
										
										
										
											2013-08-27 02:25:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | 	return cos; | 
					
						
							| 
									
										
										
										
											2011-12-06 22:55:41 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2012-01-06 00:08:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Find the index of the loop in 'poly' which references vertex, | 
					
						
							|  |  |  |  * returns -1 if not found | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2015-04-19 14:21:23 +10:00
										 |  |  | int poly_find_loop_from_vert( | 
					
						
							|  |  |  |         const MPoly *poly, const MLoop *loopstart, | 
					
						
							|  |  |  |         unsigned vert) | 
					
						
							| 
									
										
										
										
											2011-04-15 01:19:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | 	int j; | 
					
						
							|  |  |  | 	for (j = 0; j < poly->totloop; j++, loopstart++) { | 
					
						
							|  |  |  | 		if (loopstart->v == vert) | 
					
						
							|  |  |  | 			return j; | 
					
						
							| 
									
										
										
										
											2011-04-15 01:19:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | 	return -1; | 
					
						
							| 
									
										
										
										
											2004-03-20 23:59:57 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2005-07-17 21:24:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2014-03-16 03:24:05 +11:00
										 |  |  |  * Fill \a r_adj with the loop indices in \a poly adjacent to the | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  |  * vertex. Returns the index of the loop matching vertex, or -1 if the | 
					
						
							|  |  |  |  * vertex is not in \a poly | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2015-04-19 14:21:23 +10:00
										 |  |  | int poly_get_adj_loops_from_vert( | 
					
						
							| 
									
										
										
										
											2015-07-11 04:39:27 +10:00
										 |  |  |         const MPoly *poly, | 
					
						
							|  |  |  |         const MLoop *mloop, unsigned int vert, | 
					
						
							|  |  |  |         unsigned int r_adj[2]) | 
					
						
							| 
									
										
										
										
											2012-01-06 00:08:37 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | 	int corner = poly_find_loop_from_vert(poly, | 
					
						
							|  |  |  | 	                                      &mloop[poly->loopstart], | 
					
						
							|  |  |  | 	                                      vert); | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 	if (corner != -1) { | 
					
						
							| 
									
										
										
										
											2015-04-19 14:21:23 +10:00
										 |  |  | #if 0	/* unused - this loop */
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | 		const MLoop *ml = &mloop[poly->loopstart + corner]; | 
					
						
							| 
									
										
										
										
											2015-04-19 14:21:23 +10:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2013-05-10 10:07:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | 		/* vertex was found */ | 
					
						
							| 
									
										
										
										
											2014-03-16 03:24:05 +11:00
										 |  |  | 		r_adj[0] = ME_POLY_LOOP_PREV(mloop, poly, corner)->v; | 
					
						
							| 
									
										
										
										
											2015-04-19 14:21:23 +10:00
										 |  |  | 		r_adj[1] = ME_POLY_LOOP_NEXT(mloop, poly, corner)->v; | 
					
						
							| 
									
										
										
										
											2013-05-10 10:07:01 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-01-06 00:08:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | 	return corner; | 
					
						
							| 
									
										
										
										
											2013-05-10 10:07:01 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2012-01-06 00:08:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Return the index of the edge vert that is not equal to \a v. If | 
					
						
							|  |  |  |  * neither edge vertex is equal to \a v, returns -1. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int BKE_mesh_edge_other_vert(const MEdge *e, int v) | 
					
						
							| 
									
										
										
										
											2013-05-10 10:07:01 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | 	if (e->v1 == v) | 
					
						
							|  |  |  | 		return e->v2; | 
					
						
							|  |  |  | 	else if (e->v2 == v) | 
					
						
							|  |  |  | 		return e->v1; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-06-12 09:35:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | /* basic vertex data functions */ | 
					
						
							| 
									
										
										
										
											2015-07-11 04:39:27 +10:00
										 |  |  | bool BKE_mesh_minmax(const Mesh *me, float r_min[3], float r_max[3]) | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	int i = me->totvert; | 
					
						
							|  |  |  | 	MVert *mvert; | 
					
						
							|  |  |  | 	for (mvert = me->mvert; i--; mvert++) { | 
					
						
							|  |  |  | 		minmax_v3v3_v3(r_min, r_max, mvert->co); | 
					
						
							| 
									
										
										
										
											2013-06-12 09:35:02 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	return (me->totvert != 0); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-06-12 09:35:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-01 20:09:31 +10:00
										 |  |  | void BKE_mesh_transform(Mesh *me, float mat[4][4], bool do_keys) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 	MVert *mvert = me->mvert; | 
					
						
							| 
									
										
										
										
											2015-05-03 11:55:58 +02:00
										 |  |  | 	float (*lnors)[3] = CustomData_get_layer(&me->ldata, CD_NORMAL); | 
					
						
							| 
									
										
										
										
											2014-09-01 20:09:31 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < me->totvert; i++, mvert++) | 
					
						
							|  |  |  | 		mul_m4_v3(mat, mvert->co); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (do_keys && me->key) { | 
					
						
							|  |  |  | 		KeyBlock *kb; | 
					
						
							|  |  |  | 		for (kb = me->key->block.first; kb; kb = kb->next) { | 
					
						
							|  |  |  | 			float *fp = kb->data; | 
					
						
							|  |  |  | 			for (i = kb->totelem; i--; fp += 3) { | 
					
						
							|  |  |  | 				mul_m4_v3(mat, fp); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-03 11:55:58 +02:00
										 |  |  | 	/* don't update normals, caller can do this explicitly.
 | 
					
						
							|  |  |  | 	 * We do update loop normals though, those may not be auto-generated (see e.g. STL import script)! */ | 
					
						
							|  |  |  | 	if (lnors) { | 
					
						
							|  |  |  | 		float m3[3][3]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		copy_m3_m4(m3, mat); | 
					
						
							|  |  |  | 		normalize_m3(m3); | 
					
						
							|  |  |  | 		for (i = 0; i < me->totloop; i++, lnors++) { | 
					
						
							|  |  |  | 			mul_m3_v3(m3, *lnors); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-09-01 20:09:31 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | void BKE_mesh_translate(Mesh *me, const float offset[3], const bool do_keys) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i = me->totvert; | 
					
						
							|  |  |  | 	MVert *mvert; | 
					
						
							|  |  |  | 	for (mvert = me->mvert; i--; mvert++) { | 
					
						
							|  |  |  | 		add_v3_v3(mvert->co, offset); | 
					
						
							| 
									
										
										
										
											2013-05-30 18:09:19 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | 	 | 
					
						
							|  |  |  | 	if (do_keys && me->key) { | 
					
						
							|  |  |  | 		KeyBlock *kb; | 
					
						
							|  |  |  | 		for (kb = me->key->block.first; kb; kb = kb->next) { | 
					
						
							|  |  |  | 			float *fp = kb->data; | 
					
						
							|  |  |  | 			for (i = kb->totelem; i--; fp += 3) { | 
					
						
							|  |  |  | 				add_v3_v3(fp, offset); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2013-05-30 18:09:19 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2013-05-10 10:07:01 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2012-01-06 00:08:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | void BKE_mesh_ensure_navmesh(Mesh *me) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!CustomData_has_layer(&me->pdata, CD_RECAST)) { | 
					
						
							|  |  |  | 		int i; | 
					
						
							|  |  |  | 		int numFaces = me->totpoly; | 
					
						
							|  |  |  | 		int *recastData; | 
					
						
							|  |  |  | 		recastData = (int *)MEM_mallocN(numFaces * sizeof(int), __func__); | 
					
						
							|  |  |  | 		for (i = 0; i < numFaces; i++) { | 
					
						
							|  |  |  | 			recastData[i] = i + 1; | 
					
						
							| 
									
										
										
										
											2012-09-17 22:22:06 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | 		CustomData_add_layer_named(&me->pdata, CD_RECAST, CD_ASSIGN, recastData, numFaces, "recastData"); | 
					
						
							| 
									
										
										
										
											2012-01-06 00:08:37 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | void BKE_mesh_tessface_calc(Mesh *mesh) | 
					
						
							| 
									
										
										
										
											2013-05-28 14:23:07 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | 	mesh->totface = BKE_mesh_recalc_tessellation(&mesh->fdata, &mesh->ldata, &mesh->pdata, | 
					
						
							|  |  |  | 	                                             mesh->mvert, | 
					
						
							|  |  |  | 	                                             mesh->totface, mesh->totloop, mesh->totpoly, | 
					
						
							|  |  |  | 	                                             /* calc normals right after, don't copy from polys here */ | 
					
						
							|  |  |  | 	                                             false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BKE_mesh_update_customdata_pointers(mesh, true); | 
					
						
							| 
									
										
										
										
											2013-05-28 14:23:07 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | void BKE_mesh_tessface_ensure(Mesh *mesh) | 
					
						
							| 
									
										
										
										
											2005-07-17 21:24:43 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | 	if (mesh->totpoly && mesh->totface == 0) { | 
					
						
							|  |  |  | 		BKE_mesh_tessface_calc(mesh); | 
					
						
							| 
									
										
										
										
											2005-07-17 21:24:43 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2005-07-22 17:03:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-09 02:11:44 +00:00
										 |  |  | void BKE_mesh_tessface_clear(Mesh *mesh) | 
					
						
							| 
									
										
										
										
											2012-02-08 11:52:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-04-01 11:34:00 +11:00
										 |  |  | 	mesh_tessface_clear_intern(mesh, true); | 
					
						
							| 
									
										
										
										
											2012-02-08 11:52:44 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2012-05-08 14:58:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-10 04:43:31 +00:00
										 |  |  | void BKE_mesh_do_versions_cd_flag_init(Mesh *mesh) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (UNLIKELY(mesh->cd_flag)) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		MVert *mv; | 
					
						
							|  |  |  | 		MEdge *med; | 
					
						
							|  |  |  | 		int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for (mv = mesh->mvert, i = 0; i < mesh->totvert; mv++, i++) { | 
					
						
							|  |  |  | 			if (mv->bweight != 0) { | 
					
						
							|  |  |  | 				mesh->cd_flag |= ME_CDFLAG_VERT_BWEIGHT; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for (med = mesh->medge, i = 0; i < mesh->totedge; med++, i++) { | 
					
						
							|  |  |  | 			if (med->bweight != 0) { | 
					
						
							|  |  |  | 				mesh->cd_flag |= ME_CDFLAG_EDGE_BWEIGHT; | 
					
						
							|  |  |  | 				if (mesh->cd_flag & ME_CDFLAG_EDGE_CREASE) { | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if (med->crease != 0) { | 
					
						
							|  |  |  | 				mesh->cd_flag |= ME_CDFLAG_EDGE_CREASE; | 
					
						
							|  |  |  | 				if (mesh->cd_flag & ME_CDFLAG_EDGE_BWEIGHT) { | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2013-06-15 09:42:58 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /* MSelect functions (currently used in weight paint mode) */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BKE_mesh_mselect_clear(Mesh *me) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-06-15 11:02:36 +00:00
										 |  |  | 	if (me->mselect) { | 
					
						
							|  |  |  | 		MEM_freeN(me->mselect); | 
					
						
							|  |  |  | 		me->mselect = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-06-15 09:42:58 +00:00
										 |  |  | 	me->totselect = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BKE_mesh_mselect_validate(Mesh *me) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	MSelect *mselect_src, *mselect_dst; | 
					
						
							|  |  |  | 	int i_src, i_dst; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (me->totselect == 0) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	mselect_src = me->mselect; | 
					
						
							|  |  |  | 	mselect_dst = MEM_mallocN(sizeof(MSelect) * (me->totselect), "Mesh selection history"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i_src = 0, i_dst = 0; i_src < me->totselect; i_src++) { | 
					
						
							|  |  |  | 		int index = mselect_src[i_src].index; | 
					
						
							|  |  |  | 		switch (mselect_src[i_src].type) { | 
					
						
							|  |  |  | 			case ME_VSEL: | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				if (me->mvert[index].flag & SELECT) { | 
					
						
							|  |  |  | 					mselect_dst[i_dst] = mselect_src[i_src]; | 
					
						
							|  |  |  | 					i_dst++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			case ME_ESEL: | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				if (me->medge[index].flag & SELECT) { | 
					
						
							|  |  |  | 					mselect_dst[i_dst] = mselect_src[i_src]; | 
					
						
							|  |  |  | 					i_dst++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			case ME_FSEL: | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				if (me->mpoly[index].flag & SELECT) { | 
					
						
							|  |  |  | 					mselect_dst[i_dst] = mselect_src[i_src]; | 
					
						
							|  |  |  | 					i_dst++; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			default: | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				BLI_assert(0); | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	MEM_freeN(mselect_src); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (i_dst == 0) { | 
					
						
							|  |  |  | 		MEM_freeN(mselect_dst); | 
					
						
							|  |  |  | 		mselect_dst = NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if (i_dst != me->totselect) { | 
					
						
							|  |  |  | 		mselect_dst = MEM_reallocN(mselect_dst, sizeof(MSelect) * i_dst); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	me->totselect = i_dst; | 
					
						
							|  |  |  | 	me->mselect = mselect_dst; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Return the index within me->mselect, or -1 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int BKE_mesh_mselect_find(Mesh *me, int index, int type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-20 01:30:29 +10:00
										 |  |  | 	BLI_assert(ELEM(type, ME_VSEL, ME_ESEL, ME_FSEL)); | 
					
						
							| 
									
										
										
										
											2013-06-15 09:42:58 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < me->totselect; i++) { | 
					
						
							|  |  |  | 		if ((me->mselect[i].index == index) && | 
					
						
							| 
									
										
										
										
											2014-01-12 22:05:24 +11:00
										 |  |  | 		    (me->mselect[i].type == type)) | 
					
						
							| 
									
										
										
										
											2013-06-15 09:42:58 +00:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 			return i; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Return The index of the active element. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int BKE_mesh_mselect_active_get(Mesh *me, int type) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-07-20 01:30:29 +10:00
										 |  |  | 	BLI_assert(ELEM(type, ME_VSEL, ME_ESEL, ME_FSEL)); | 
					
						
							| 
									
										
										
										
											2013-06-15 09:42:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-30 22:04:03 +00:00
										 |  |  | 	if (me->totselect) { | 
					
						
							| 
									
										
										
										
											2013-06-15 09:42:58 +00:00
										 |  |  | 		if (me->mselect[me->totselect - 1].type == type) { | 
					
						
							|  |  |  | 			return me->mselect[me->totselect - 1].index; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BKE_mesh_mselect_active_set(Mesh *me, int index, int type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const int msel_index = BKE_mesh_mselect_find(me, index, type); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (msel_index == -1) { | 
					
						
							|  |  |  | 		/* add to the end */ | 
					
						
							|  |  |  | 		me->mselect = MEM_reallocN(me->mselect, sizeof(MSelect) * (me->totselect + 1)); | 
					
						
							|  |  |  | 		me->mselect[me->totselect].index = index; | 
					
						
							|  |  |  | 		me->mselect[me->totselect].type  = type; | 
					
						
							|  |  |  | 		me->totselect++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if (msel_index != me->totselect - 1) { | 
					
						
							|  |  |  | 		/* move to the end */ | 
					
						
							|  |  |  | 		SWAP(MSelect, me->mselect[msel_index], me->mselect[me->totselect - 1]); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BLI_assert((me->mselect[me->totselect - 1].index == index) && | 
					
						
							|  |  |  | 	           (me->mselect[me->totselect - 1].type  == type)); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-20 12:41:31 +05:00
										 |  |  | void BKE_mesh_calc_normals_split(Mesh *mesh) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	float (*r_loopnors)[3]; | 
					
						
							|  |  |  | 	float (*polynors)[3]; | 
					
						
							|  |  |  | 	short (*clnors)[2] = NULL; | 
					
						
							|  |  |  | 	bool free_polynors = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (CustomData_has_layer(&mesh->ldata, CD_NORMAL)) { | 
					
						
							|  |  |  | 		r_loopnors = CustomData_get_layer(&mesh->ldata, CD_NORMAL); | 
					
						
							|  |  |  | 		memset(r_loopnors, 0, sizeof(float[3]) * mesh->totloop); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		r_loopnors = CustomData_add_layer(&mesh->ldata, CD_NORMAL, CD_CALLOC, NULL, mesh->totloop); | 
					
						
							|  |  |  | 		CustomData_set_layer_flag(&mesh->ldata, CD_NORMAL, CD_FLAG_TEMPORARY); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* may be NULL */ | 
					
						
							|  |  |  | 	clnors = CustomData_get_layer(&mesh->ldata, CD_CUSTOMLOOPNORMAL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (CustomData_has_layer(&mesh->pdata, CD_NORMAL)) { | 
					
						
							|  |  |  | 		/* This assume that layer is always up to date, not sure this is the case (esp. in Edit mode?)... */ | 
					
						
							|  |  |  | 		polynors = CustomData_get_layer(&mesh->pdata, CD_NORMAL); | 
					
						
							|  |  |  | 		free_polynors = false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		polynors = MEM_mallocN(sizeof(float[3]) * mesh->totpoly, __func__); | 
					
						
							| 
									
										
										
										
											2015-10-12 20:12:55 +02:00
										 |  |  | 		BKE_mesh_calc_normals_poly( | 
					
						
							|  |  |  | 		            mesh->mvert, NULL, mesh->totvert, | 
					
						
							|  |  |  | 		            mesh->mloop, mesh->mpoly, mesh->totloop, mesh->totpoly, polynors, false); | 
					
						
							| 
									
										
										
										
											2015-03-20 12:41:31 +05:00
										 |  |  | 		free_polynors = true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	BKE_mesh_normals_loop_split( | 
					
						
							|  |  |  | 	        mesh->mvert, mesh->totvert, mesh->medge, mesh->totedge, | 
					
						
							|  |  |  | 	        mesh->mloop, r_loopnors, mesh->totloop, mesh->mpoly, (const float (*)[3])polynors, mesh->totpoly, | 
					
						
							|  |  |  | 	        (mesh->flag & ME_AUTOSMOOTH) != 0, mesh->smoothresh, NULL, clnors, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (free_polynors) { | 
					
						
							|  |  |  | 		MEM_freeN(polynors); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | /* Split faces helper functions. */ | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | enum { | 
					
						
							|  |  |  | 	/* Vertex is adjacent to some loop which normal is different,
 | 
					
						
							|  |  |  | 	 * hence split of this vertex is required. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	SPLIT_VERT_NEED_SPLIT = (1 << 0), | 
					
						
							|  |  |  | 	/* Original vertex was already re-used by split logic. */ | 
					
						
							|  |  |  | 	SPLIT_VERT_REUSED     = (1 << 1), | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | enum { | 
					
						
							|  |  |  | 	/* Edge is adjacent to any of vertex tagged for split.
 | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	SPLIT_EDGE_NEED_SPLIT = (1 << 0), | 
					
						
							|  |  |  | 	/* Original edge was already re-used by split logic. */ | 
					
						
							|  |  |  | 	SPLIT_EDGE_REUSED     = (1 << 1), | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | /* Tag vertices which normals are not equal to any adjacent loop
 | 
					
						
							|  |  |  |  * and hence split on that vertex is required. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Returns truth if any of vertex needs to be split. | 
					
						
							| 
									
										
										
										
											2015-03-20 12:41:31 +05:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | static bool split_faces_tag_verts(const Mesh *mesh, uchar *vert_flags) | 
					
						
							| 
									
										
										
										
											2015-03-20 12:41:31 +05:00
										 |  |  | { | 
					
						
							|  |  |  | 	const int num_polys = mesh->totpoly; | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 	const MVert *mvert = mesh->mvert; | 
					
						
							|  |  |  | 	const MLoop *mloop = mesh->mloop; | 
					
						
							|  |  |  | 	const MPoly *mpoly = mesh->mpoly; | 
					
						
							|  |  |  | 	float (*lnors)[3] = CustomData_get_layer(&mesh->ldata, CD_NORMAL); | 
					
						
							|  |  |  | 	bool has_split_verts = false; | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 	for (int poly = 0; poly < num_polys; poly++) { | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 		const MPoly *mp = &mpoly[poly]; | 
					
						
							| 
									
										
										
										
											2017-02-14 15:30:49 +01:00
										 |  |  | 		for (int loop = 0; loop < mp->totloop; loop++) { | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 			const MLoop *ml = &mloop[mp->loopstart + loop]; | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 			const MVert *mv = &mvert[ml->v]; | 
					
						
							| 
									
										
										
										
											2015-03-20 12:41:31 +05:00
										 |  |  | 			float vn[3]; | 
					
						
							|  |  |  | 			normal_short_to_float_v3(vn, mv->no); | 
					
						
							| 
									
										
										
										
											2017-02-15 15:21:15 +01:00
										 |  |  | 			if (len_squared_v3v3(vn, lnors[mp->loopstart + loop]) > FLT_EPSILON) { | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 				vert_flags[ml->v] |= SPLIT_VERT_NEED_SPLIT; | 
					
						
							|  |  |  | 				has_split_verts = true; | 
					
						
							| 
									
										
										
										
											2015-03-20 12:41:31 +05:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 	return has_split_verts; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Count number of new vertices to be added.
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Note that one of the loop where split is required will re-use | 
					
						
							|  |  |  |  * it's vertex in order to avoid creation of loose vertices. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int split_faces_count_new_verts(const Mesh *mesh, uchar *vert_flags) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const int num_polys = mesh->totpoly; | 
					
						
							|  |  |  | 	const MLoop *mloop = mesh->mloop; | 
					
						
							|  |  |  | 	const MPoly *mpoly = mesh->mpoly; | 
					
						
							|  |  |  | 	int num_new_verts = 0; | 
					
						
							|  |  |  | 	for (int poly = 0; poly < num_polys; poly++) { | 
					
						
							|  |  |  | 		const MPoly *mp = &mpoly[poly]; | 
					
						
							|  |  |  | 		for (int loop = 0; loop < mp->totloop; loop++) { | 
					
						
							|  |  |  | 			const MLoop *ml = &mloop[mp->loopstart + loop]; | 
					
						
							|  |  |  | 			if (vert_flags[ml->v] & SPLIT_VERT_NEED_SPLIT) { | 
					
						
							|  |  |  | 				if (vert_flags[ml->v] & SPLIT_VERT_REUSED) { | 
					
						
							|  |  |  | 					++num_new_verts; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					vert_flags[ml->v] |= SPLIT_VERT_REUSED; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-03-20 12:41:31 +05:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 	return num_new_verts; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Tag edges which are adjacent to at least one vertex tagged for split. */ | 
					
						
							|  |  |  | static void split_faces_tag_edges(Mesh *mesh, | 
					
						
							|  |  |  |                                   const uchar *vert_flags, | 
					
						
							|  |  |  |                                   uchar *edge_flags) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const int num_polys = mesh->totpoly; | 
					
						
							|  |  |  | 	const MLoop *mloop = mesh->mloop; | 
					
						
							|  |  |  | 	const MPoly *mpoly = mesh->mpoly; | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 	for (int poly = 0; poly < num_polys; poly++) { | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 		const MPoly *mp = &mpoly[poly]; | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 		int loop_prev = mp->totloop - 1; | 
					
						
							|  |  |  | 		for (int loop = 0; loop < mp->totloop; loop++) { | 
					
						
							|  |  |  | 			const int poly_loop_prev = mp->loopstart + loop_prev; | 
					
						
							|  |  |  | 			const MLoop *ml = &mloop[mp->loopstart + loop]; | 
					
						
							|  |  |  | 			const MLoop *ml_prev = &mloop[poly_loop_prev]; | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 			const int mv_flag = vert_flags[ml->v]; | 
					
						
							|  |  |  | 			const int mv_prev_flag = vert_flags[ml_prev->v]; | 
					
						
							|  |  |  | 			bool need_split = false; | 
					
						
							|  |  |  | 			if (mv_flag & SPLIT_VERT_NEED_SPLIT) { | 
					
						
							|  |  |  | 				if (mv_prev_flag & SPLIT_VERT_NEED_SPLIT) { | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 					/* Create new edge between twp split vertices. */ | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 					need_split = true; | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					/* Create new edge from existing vertex to a split one. */ | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 					need_split = true; | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 			else if (mv_prev_flag & SPLIT_VERT_NEED_SPLIT) { | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 				/* Create new edge from split vertex to existing one. */ | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 				need_split = true; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if (need_split) { | 
					
						
							|  |  |  | 				edge_flags[ml_prev->e] |= SPLIT_EDGE_NEED_SPLIT; | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			loop_prev = loop; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Count number of new edges to be added.
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Note that one of the loop where split is required will re-use | 
					
						
							|  |  |  |  * it's edge in order to avoid creation of loose edges. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static int split_faces_count_new_edges(const Mesh *mesh, uchar *edge_flags) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const int num_polys = mesh->totpoly; | 
					
						
							|  |  |  | 	const MLoop *mloop = mesh->mloop; | 
					
						
							|  |  |  | 	const MPoly *mpoly = mesh->mpoly; | 
					
						
							|  |  |  | 	int num_new_edges = 0; | 
					
						
							|  |  |  | 	for (int poly = 0; poly < num_polys; poly++) { | 
					
						
							|  |  |  | 		const MPoly *mp = &mpoly[poly]; | 
					
						
							|  |  |  | 		for (int loop = 0; loop < mp->totloop; loop++) { | 
					
						
							|  |  |  | 			const MLoop *ml = &mloop[mp->loopstart + loop]; | 
					
						
							|  |  |  | 			if (edge_flags[ml->e] & SPLIT_EDGE_NEED_SPLIT) { | 
					
						
							|  |  |  | 				if (edge_flags[ml->e] & SPLIT_EDGE_REUSED) { | 
					
						
							|  |  |  | 					++num_new_edges; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							|  |  |  | 					edge_flags[ml->e] |= SPLIT_EDGE_REUSED; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return num_new_edges; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Perform actual split of vertices.
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * NOTE: Will leave edges in inconsistent state. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void split_faces_split_verts(Mesh *mesh, | 
					
						
							|  |  |  |                                     const int num_new_verts, | 
					
						
							|  |  |  |                                     uchar *vert_flags) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const int num_verts = mesh->totvert - num_new_verts; | 
					
						
							|  |  |  | 	const int num_polys = mesh->totpoly; | 
					
						
							|  |  |  | 	MVert *mvert = mesh->mvert; | 
					
						
							|  |  |  | 	MLoop *mloop = mesh->mloop; | 
					
						
							|  |  |  | 	MPoly *mpoly = mesh->mpoly; | 
					
						
							|  |  |  | 	const float (*lnors)[3] = CustomData_get_layer(&mesh->ldata, CD_NORMAL); | 
					
						
							|  |  |  | 	int num_added_verts = 0; | 
					
						
							|  |  |  | 	/* Clear reused flag, we need it again. */ | 
					
						
							|  |  |  | 	for (int i = 0; i < num_verts; ++i) { | 
					
						
							|  |  |  | 		vert_flags[i] &= ~SPLIT_VERT_REUSED; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 	for (int poly = 0; poly < num_polys; poly++) { | 
					
						
							| 
									
										
										
										
											2015-03-20 12:41:31 +05:00
										 |  |  | 		MPoly *mp = &mpoly[poly]; | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 		/* First we split all vertices to get proper flag whether they are
 | 
					
						
							|  |  |  | 		 * split or not for all of them before handling edges. | 
					
						
							|  |  |  | 		 */ | 
					
						
							| 
									
										
										
										
											2017-02-14 15:30:49 +01:00
										 |  |  | 		for (int loop = 0; loop < mp->totloop; loop++) { | 
					
						
							| 
									
										
										
										
											2015-03-20 12:41:31 +05:00
										 |  |  | 			int poly_loop = mp->loopstart + loop; | 
					
						
							|  |  |  | 			MLoop *ml = &mloop[poly_loop]; | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 			if (vert_flags[ml->v] & SPLIT_VERT_NEED_SPLIT) { | 
					
						
							|  |  |  | 				if ((vert_flags[ml->v] & SPLIT_VERT_REUSED) == 0) { | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 					/* Ignore first split on vertex, re-use it instead. */ | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 					vert_flags[ml->v] |= SPLIT_VERT_REUSED; | 
					
						
							| 
									
										
										
										
											2017-02-20 11:47:43 +01:00
										 |  |  | 					normal_float_to_short_v3(mvert[ml->v].no, lnors[poly_loop]); | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 					continue; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 				/* Create new vertex. */ | 
					
						
							|  |  |  | 				int new_vert = num_verts + num_added_verts; | 
					
						
							| 
									
										
										
										
											2015-03-20 12:41:31 +05:00
										 |  |  | 				CustomData_copy_data(&mesh->vdata, &mesh->vdata, | 
					
						
							|  |  |  | 				                     ml->v, new_vert, 1); | 
					
						
							|  |  |  | 				normal_float_to_short_v3(mvert[new_vert].no, | 
					
						
							|  |  |  | 				                         lnors[poly_loop]); | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 				ml->v = new_vert; | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 				num_added_verts++; | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-02-15 20:59:55 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Perform actual split of edges.
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * NOTE: Will correct all edges. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void split_faces_split_edges(Mesh *mesh, | 
					
						
							|  |  |  |                                     const int num_new_edges, | 
					
						
							|  |  |  |                                     uchar *edge_flags) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const int num_edges = mesh->totedge - num_new_edges; | 
					
						
							|  |  |  | 	const int num_polys = mesh->totpoly; | 
					
						
							|  |  |  | 	MEdge *medge = mesh->medge; | 
					
						
							|  |  |  | 	MLoop *mloop = mesh->mloop; | 
					
						
							|  |  |  | 	MPoly *mpoly = mesh->mpoly; | 
					
						
							|  |  |  | 	int num_added_edges = 0; | 
					
						
							|  |  |  | 	/* Clear reused flag, we need it again. */ | 
					
						
							|  |  |  | 	for (int i = 0; i < num_edges; ++i) { | 
					
						
							|  |  |  | 		edge_flags[i] &= ~SPLIT_EDGE_REUSED; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-15 20:59:55 +01:00
										 |  |  | 	for (int poly = 0; poly < num_polys; poly++) { | 
					
						
							|  |  |  | 		MPoly *mp = &mpoly[poly]; | 
					
						
							|  |  |  | 		for (int loop = 0, loop_prev = mp->totloop - 1; loop < mp->totloop; loop++) { | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 			const int poly_loop_prev = mp->loopstart + loop_prev; | 
					
						
							|  |  |  | 			const MLoop *ml = &mloop[mp->loopstart + loop]; | 
					
						
							|  |  |  | 			MLoop *ml_prev = &mloop[poly_loop_prev]; | 
					
						
							|  |  |  | 			MEdge *me_prev = &medge[ml_prev->e]; | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 			if (edge_flags[ml_prev->e] & SPLIT_EDGE_NEED_SPLIT) { | 
					
						
							|  |  |  | 				if ((edge_flags[ml_prev->e] & SPLIT_EDGE_REUSED) == 0) { | 
					
						
							|  |  |  | 					edge_flags[ml_prev->e] |= SPLIT_EDGE_REUSED; | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 					me_prev->v1 = ml_prev->v; | 
					
						
							|  |  |  | 					me_prev->v2 = ml->v; | 
					
						
							| 
									
										
										
										
											2017-02-15 12:33:04 +01:00
										 |  |  | 				} | 
					
						
							|  |  |  | 				else { | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 					const int index = num_edges + num_added_edges; | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 					CustomData_copy_data(&mesh->edata, &mesh->edata, | 
					
						
							|  |  |  | 					                     ml_prev->e, index, 1); | 
					
						
							|  |  |  | 					MEdge *me_new = &medge[index]; | 
					
						
							|  |  |  | 					me_new->v1 = ml_prev->v; | 
					
						
							|  |  |  | 					me_new->v2 = ml->v; | 
					
						
							|  |  |  | 					ml_prev->e = index; | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | 					num_added_edges++; | 
					
						
							| 
									
										
										
										
											2017-02-14 17:02:22 +01:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | 			loop_prev = loop; | 
					
						
							| 
									
										
										
										
											2017-02-14 17:02:22 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-03-20 12:41:31 +05:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Split faces based on the edge angle.
 | 
					
						
							|  |  |  |  * Matches behavior of face splitting in render engines. | 
					
						
							| 
									
										
										
										
											2017-02-20 11:56:02 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  * NOTE: Will leave CD_NORMAL loop data layer which is | 
					
						
							|  |  |  |  * used by render engines to set shading up. | 
					
						
							| 
									
										
										
										
											2017-02-15 23:09:31 +01:00
										 |  |  |  */ | 
					
						
							|  |  |  | void BKE_mesh_split_faces(Mesh *mesh) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const int num_verts = mesh->totvert; | 
					
						
							|  |  |  | 	const int num_edges = mesh->totedge; | 
					
						
							|  |  |  | 	const int num_polys = mesh->totpoly; | 
					
						
							|  |  |  | 	if ((mesh->flag & ME_AUTOSMOOTH) == 0) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (num_polys == 0) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	BKE_mesh_tessface_clear(mesh); | 
					
						
							|  |  |  | 	/* Compute loop normals if needed. */ | 
					
						
							|  |  |  | 	if (!CustomData_has_layer(&mesh->ldata, CD_NORMAL)) { | 
					
						
							|  |  |  | 		BKE_mesh_calc_normals_split(mesh); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* Runtime flags. */ | 
					
						
							|  |  |  | 	uchar *vert_flags = MEM_callocN(sizeof(*vert_flags) * num_verts, | 
					
						
							|  |  |  | 	                                "split faces vert flags"); | 
					
						
							|  |  |  | 	/* Tag vertces and check whether anything is tagged. */ | 
					
						
							|  |  |  | 	if (!split_faces_tag_verts(mesh, vert_flags)) { | 
					
						
							|  |  |  | 		/* No new vertices to be split added, can do early exit. */ | 
					
						
							|  |  |  | 		MEM_freeN(vert_flags); | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* Flush vertex flags to edges. */ | 
					
						
							|  |  |  | 	uchar *edge_flags = MEM_callocN(sizeof(*edge_flags) * num_edges, | 
					
						
							|  |  |  | 	                                "split faces edge flags"); | 
					
						
							|  |  |  | 	split_faces_tag_edges(mesh, vert_flags, edge_flags); | 
					
						
							|  |  |  | 	/* Count amount of new geometry. */ | 
					
						
							|  |  |  | 	int num_new_verts = split_faces_count_new_verts(mesh, vert_flags); | 
					
						
							|  |  |  | 	int num_new_edges =  split_faces_count_new_edges(mesh, edge_flags); | 
					
						
							|  |  |  | 	/* Reallocate all vert and edge related data. */ | 
					
						
							|  |  |  | 	mesh->totvert += num_new_verts; | 
					
						
							|  |  |  | 	mesh->totedge += num_new_edges; | 
					
						
							|  |  |  | 	CustomData_realloc(&mesh->vdata, mesh->totvert); | 
					
						
							|  |  |  | 	CustomData_realloc(&mesh->edata, mesh->totedge); | 
					
						
							|  |  |  | 	/* Update pointers to a newly allocated memory. */ | 
					
						
							|  |  |  | 	BKE_mesh_update_customdata_pointers(mesh, false); | 
					
						
							|  |  |  | 	/* Perform actual split of vertices and adjacent edges. */ | 
					
						
							|  |  |  | 	split_faces_split_verts(mesh, num_new_verts, vert_flags); | 
					
						
							|  |  |  | 	split_faces_split_edges(mesh, num_new_edges, edge_flags); | 
					
						
							|  |  |  | 	MEM_freeN(vert_flags); | 
					
						
							|  |  |  | 	MEM_freeN(edge_flags); | 
					
						
							| 
									
										
										
										
											2017-02-15 13:56:50 +01:00
										 |  |  | #ifdef VALIDATE_MESH
 | 
					
						
							|  |  |  | 	BKE_mesh_validate(mesh, true, true); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-03-20 12:41:31 +05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | /* settings: 1 - preview, 2 - render */ | 
					
						
							|  |  |  | Mesh *BKE_mesh_new_from_object( | 
					
						
							|  |  |  |         Main *bmain, Scene *sce, Object *ob, | 
					
						
							|  |  |  |         int apply_modifiers, int settings, int calc_tessface, int calc_undeformed) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Mesh *tmpmesh; | 
					
						
							|  |  |  | 	Curve *tmpcu = NULL, *copycu; | 
					
						
							| 
									
										
										
										
											2016-08-30 21:08:11 +02:00
										 |  |  | 	int i; | 
					
						
							|  |  |  | 	const bool render = (settings == eModifierMode_Render); | 
					
						
							|  |  |  | 	const bool cage = !apply_modifiers; | 
					
						
							| 
									
										
										
										
											2016-10-03 17:38:14 +02:00
										 |  |  | 	bool do_mat_id_data_us = true; | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* perform the mesh extraction based on type */ | 
					
						
							|  |  |  | 	switch (ob->type) { | 
					
						
							|  |  |  | 		case OB_FONT: | 
					
						
							|  |  |  | 		case OB_CURVE: | 
					
						
							|  |  |  | 		case OB_SURF: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			ListBase dispbase = {NULL, NULL}; | 
					
						
							|  |  |  | 			DerivedMesh *derivedFinal = NULL; | 
					
						
							|  |  |  | 			int uv_from_orco; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* copies object and modifiers (but not the data) */ | 
					
						
							| 
									
										
										
										
											2016-07-17 03:03:22 +10:00
										 |  |  | 			Object *tmpobj = BKE_object_copy_ex(bmain, ob, true); | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 			tmpcu = (Curve *)tmpobj->data; | 
					
						
							| 
									
										
										
										
											2015-11-09 19:47:10 +01:00
										 |  |  | 			id_us_min(&tmpcu->id); | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-11 12:00:57 +05:00
										 |  |  | 			/* Copy cached display list, it might be needed by the stack evaluation.
 | 
					
						
							|  |  |  | 			 * Ideally stack should be able to use render-time display list, but doing | 
					
						
							|  |  |  | 			 * so is quite tricky and not safe so close to the release. | 
					
						
							|  |  |  | 			 * | 
					
						
							|  |  |  | 			 * TODO(sergey): Look into more proper solution. | 
					
						
							|  |  |  | 			 */ | 
					
						
							|  |  |  | 			if (ob->curve_cache != NULL) { | 
					
						
							|  |  |  | 				if (tmpobj->curve_cache == NULL) { | 
					
						
							|  |  |  | 					tmpobj->curve_cache = MEM_callocN(sizeof(CurveCache), "CurveCache for curve types"); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				BKE_displist_copy(&tmpobj->curve_cache->disp, &ob->curve_cache->disp); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 			/* if getting the original caged mesh, delete object modifiers */ | 
					
						
							|  |  |  | 			if (cage) | 
					
						
							|  |  |  | 				BKE_object_free_modifiers(tmpobj); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* copies the data */ | 
					
						
							| 
									
										
										
										
											2016-07-10 14:52:00 +02:00
										 |  |  | 			copycu = tmpobj->data = BKE_curve_copy(bmain, (Curve *) ob->data); | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			/* temporarily set edit so we get updates from edit mode, but
 | 
					
						
							|  |  |  | 			 * also because for text datablocks copying it while in edit | 
					
						
							|  |  |  | 			 * mode gives invalid data structures */ | 
					
						
							|  |  |  | 			copycu->editfont = tmpcu->editfont; | 
					
						
							|  |  |  | 			copycu->editnurb = tmpcu->editnurb; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* get updated display list, and convert to a mesh */ | 
					
						
							|  |  |  | 			BKE_displist_make_curveTypes_forRender(sce, tmpobj, &dispbase, &derivedFinal, false, render); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			copycu->editfont = NULL; | 
					
						
							|  |  |  | 			copycu->editnurb = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			tmpobj->derivedFinal = derivedFinal; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* convert object type to mesh */ | 
					
						
							|  |  |  | 			uv_from_orco = (tmpcu->flag & CU_UV_ORCO) != 0; | 
					
						
							|  |  |  | 			BKE_mesh_from_nurbs_displist(tmpobj, &dispbase, uv_from_orco); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			tmpmesh = tmpobj->data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			BKE_displist_free(&dispbase); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* BKE_mesh_from_nurbs changes the type to a mesh, check it worked.
 | 
					
						
							|  |  |  | 			 * if it didn't the curve did not have any segments or otherwise  | 
					
						
							|  |  |  | 			 * would have generated an empty mesh */ | 
					
						
							|  |  |  | 			if (tmpobj->type != OB_MESH) { | 
					
						
							| 
									
										
										
										
											2014-06-24 20:43:08 +06:00
										 |  |  | 				BKE_libblock_free_us(bmain, tmpobj); | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 				return NULL; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			BKE_mesh_texspace_copy_from_object(tmpmesh, ob); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			BKE_libblock_free_us(bmain, tmpobj); | 
					
						
							| 
									
										
										
										
											2016-08-30 20:36:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			/* XXX The curve to mesh conversion is convoluted... But essentially, BKE_mesh_from_nurbs_displist()
 | 
					
						
							|  |  |  | 			 *     already transfers the ownership of materials from the temp copy of the Curve ID to the new | 
					
						
							|  |  |  | 			 *     Mesh ID, so we do not want to increase materials' usercount later. */ | 
					
						
							| 
									
										
										
										
											2016-10-03 17:38:14 +02:00
										 |  |  | 			do_mat_id_data_us = false; | 
					
						
							| 
									
										
										
										
											2016-08-30 20:36:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case OB_MBALL: | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			/* metaballs don't have modifiers, so just convert to mesh */ | 
					
						
							|  |  |  | 			Object *basis_ob = BKE_mball_basis_find(sce, ob); | 
					
						
							|  |  |  | 			/* todo, re-generatre for render-res */ | 
					
						
							|  |  |  | 			/* metaball_polygonize(scene, ob) */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (ob != basis_ob) | 
					
						
							|  |  |  | 				return NULL;  /* only do basis metaball */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			tmpmesh = BKE_mesh_add(bmain, "Mesh"); | 
					
						
							|  |  |  | 			/* BKE_mesh_add gives us a user count we don't need */ | 
					
						
							| 
									
										
										
										
											2015-11-09 19:47:10 +01:00
										 |  |  | 			id_us_min(&tmpmesh->id); | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (render) { | 
					
						
							|  |  |  | 				ListBase disp = {NULL, NULL}; | 
					
						
							|  |  |  | 				/* TODO(sergey): This is gonna to work for until EvaluationContext
 | 
					
						
							|  |  |  | 				 *               only contains for_render flag. As soon as CoW is | 
					
						
							|  |  |  | 				 *               implemented, this is to be rethinked. | 
					
						
							|  |  |  | 				 */ | 
					
						
							| 
									
										
											  
											
												Depsgraph: New dependency graph integration commit
This commit integrates the work done so far on the new dependency graph system,
where goal was to replace legacy depsgraph with the new one, supporting loads of
neat features like:
- More granular dependency relation nature, which solves issues with fake cycles
  in the dependencies.
- Move towards all-animatable, by better integration of drivers into the system.
- Lay down some basis for upcoming copy-on-write, overrides and so on.
The new system is living side-by-side with the previous one and disabled by
default, so nothing will become suddenly broken. The way to enable new depsgraph
is to pass `--new-depsgraph` command line argument.
It's a bit early to consider the system production-ready, there are some TODOs
and issues were discovered during the merge period, they'll be addressed ASAP.
But it's important to merge, because it's the only way to attract artists to
really start testing this system.
There are number of assorted documents related on the design of the new system:
* http://wiki.blender.org/index.php/User:Aligorith/GSoC2013_Depsgraph#Design_Documents
* http://wiki.blender.org/index.php/User:Nazg-gul/DependencyGraph
There are also some user-related information online:
* http://code.blender.org/2015/02/blender-dependency-graph-branch-for-users/
* http://code.blender.org/2015/03/more-dependency-graph-tricks/
Kudos to everyone who was involved into the project:
- Joshua "Aligorith" Leung -- design specification, initial code
- Lukas "lukas_t" Toenne -- integrating code into blender, with further fixes
- Sergey "Sergey" "Sharybin" -- some mocking around, trying to wrap up the
  project and so
- Bassam "slikdigit" Kurdali -- stressing the new system, reporting all the
  issues and recording/writing documentation.
- Everyone else who i forgot to mention here :)
											
										 
											2015-05-12 15:05:57 +05:00
										 |  |  | 				EvaluationContext eval_ctx; | 
					
						
							|  |  |  | 				DEG_evaluation_context_init(&eval_ctx, DAG_EVAL_RENDER); | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 				BKE_displist_make_mball_forRender(&eval_ctx, sce, ob, &disp); | 
					
						
							|  |  |  | 				BKE_mesh_from_metaball(&disp, tmpmesh); | 
					
						
							|  |  |  | 				BKE_displist_free(&disp); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else { | 
					
						
							|  |  |  | 				ListBase disp = {NULL, NULL}; | 
					
						
							|  |  |  | 				if (ob->curve_cache) { | 
					
						
							|  |  |  | 					disp = ob->curve_cache->disp; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				BKE_mesh_from_metaball(&disp, tmpmesh); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			BKE_mesh_texspace_copy_from_object(tmpmesh, ob); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		case OB_MESH: | 
					
						
							|  |  |  | 			/* copies object and modifiers (but not the data) */ | 
					
						
							|  |  |  | 			if (cage) { | 
					
						
							|  |  |  | 				/* copies the data */ | 
					
						
							| 
									
										
										
										
											2016-07-10 14:52:00 +02:00
										 |  |  | 				tmpmesh = BKE_mesh_copy(bmain, ob->data); | 
					
						
							| 
									
										
										
										
											2016-08-30 20:36:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				/* XXX BKE_mesh_copy() already handles materials usercount. */ | 
					
						
							| 
									
										
										
										
											2016-10-03 17:38:14 +02:00
										 |  |  | 				do_mat_id_data_us = false; | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-08-30 20:36:22 +02:00
										 |  |  | 			/* if not getting the original caged mesh, get final derived mesh */ | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 			else { | 
					
						
							|  |  |  | 				/* Make a dummy mesh, saves copying */ | 
					
						
							|  |  |  | 				DerivedMesh *dm; | 
					
						
							|  |  |  | 				/* CustomDataMask mask = CD_MASK_BAREMESH|CD_MASK_MTFACE|CD_MASK_MCOL; */ | 
					
						
							|  |  |  | 				CustomDataMask mask = CD_MASK_MESH; /* this seems more suitable, exporter,
 | 
					
						
							| 
									
										
										
										
											2016-09-09 13:08:21 +02:00
										 |  |  | 				                                     * for example, needs CD_MASK_MDEFORMVERT */ | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				if (calc_undeformed) | 
					
						
							|  |  |  | 					mask |= CD_MASK_ORCO; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				/* Write the display mesh into the dummy mesh */ | 
					
						
							|  |  |  | 				if (render) | 
					
						
							|  |  |  | 					dm = mesh_create_derived_render(sce, ob, mask); | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					dm = mesh_create_derived_view(sce, ob, mask); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				tmpmesh = BKE_mesh_add(bmain, "Mesh"); | 
					
						
							| 
									
										
										
										
											2015-02-19 00:00:23 +05:00
										 |  |  | 				DM_to_mesh(dm, tmpmesh, ob, mask, true); | 
					
						
							| 
									
										
										
										
											2017-01-11 16:04:39 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				/* Copy autosmooth settings from original mesh. */ | 
					
						
							|  |  |  | 				Mesh *me = (Mesh *)ob->data; | 
					
						
							|  |  |  | 				tmpmesh->flag |= (me->flag & ME_AUTOSMOOTH); | 
					
						
							|  |  |  | 				tmpmesh->smoothresh = me->smoothresh; | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* BKE_mesh_add/copy gives us a user count we don't need */ | 
					
						
							| 
									
										
										
										
											2015-11-09 19:47:10 +01:00
										 |  |  | 			id_us_min(&tmpmesh->id); | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			/* "Object does not have geometry data") */ | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Copy materials to new mesh */ | 
					
						
							|  |  |  | 	switch (ob->type) { | 
					
						
							|  |  |  | 		case OB_SURF: | 
					
						
							|  |  |  | 		case OB_FONT: | 
					
						
							|  |  |  | 		case OB_CURVE: | 
					
						
							|  |  |  | 			tmpmesh->totcol = tmpcu->totcol; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* free old material list (if it exists) and adjust user counts */ | 
					
						
							|  |  |  | 			if (tmpcu->mat) { | 
					
						
							|  |  |  | 				for (i = tmpcu->totcol; i-- > 0; ) { | 
					
						
							|  |  |  | 					/* are we an object material or data based? */ | 
					
						
							| 
									
										
										
										
											2016-10-11 11:54:04 +02:00
										 |  |  | 					tmpmesh->mat[i] = give_current_material(ob, i + 1); | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 11:54:04 +02:00
										 |  |  | 					if (((ob->matbits && ob->matbits[i]) || do_mat_id_data_us)  && tmpmesh->mat[i]) { | 
					
						
							| 
									
										
										
										
											2015-11-09 19:47:10 +01:00
										 |  |  | 						id_us_plus(&tmpmesh->mat[i]->id); | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		case OB_MBALL: | 
					
						
							| 
									
										
										
										
											2016-07-20 16:29:54 +02:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 			MetaBall *tmpmb = (MetaBall *)ob->data; | 
					
						
							|  |  |  | 			tmpmesh->mat = MEM_dupallocN(tmpmb->mat); | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 			tmpmesh->totcol = tmpmb->totcol; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			/* free old material list (if it exists) and adjust user counts */ | 
					
						
							|  |  |  | 			if (tmpmb->mat) { | 
					
						
							|  |  |  | 				for (i = tmpmb->totcol; i-- > 0; ) { | 
					
						
							| 
									
										
										
										
											2016-07-20 16:29:54 +02:00
										 |  |  | 					/* are we an object material or data based? */ | 
					
						
							| 
									
										
										
										
											2016-10-11 11:54:04 +02:00
										 |  |  | 					tmpmesh->mat[i] = give_current_material(ob, i + 1); | 
					
						
							| 
									
										
										
										
											2016-07-20 16:29:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-24 10:06:00 +02:00
										 |  |  | 					if (((ob->matbits && ob->matbits[i]) || do_mat_id_data_us) && tmpmesh->mat[i]) { | 
					
						
							| 
									
										
										
										
											2016-07-20 16:29:54 +02:00
										 |  |  | 						id_us_plus(&tmpmesh->mat[i]->id); | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2016-07-20 16:29:54 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		case OB_MESH: | 
					
						
							|  |  |  | 			if (!cage) { | 
					
						
							|  |  |  | 				Mesh *origmesh = ob->data; | 
					
						
							|  |  |  | 				tmpmesh->flag = origmesh->flag; | 
					
						
							|  |  |  | 				tmpmesh->mat = MEM_dupallocN(origmesh->mat); | 
					
						
							|  |  |  | 				tmpmesh->totcol = origmesh->totcol; | 
					
						
							|  |  |  | 				tmpmesh->smoothresh = origmesh->smoothresh; | 
					
						
							|  |  |  | 				if (origmesh->mat) { | 
					
						
							|  |  |  | 					for (i = origmesh->totcol; i-- > 0; ) { | 
					
						
							|  |  |  | 						/* are we an object material or data based? */ | 
					
						
							| 
									
										
										
										
											2016-10-11 11:54:04 +02:00
										 |  |  | 						tmpmesh->mat[i] = give_current_material(ob, i + 1); | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-11 11:54:04 +02:00
										 |  |  | 						if (((ob->matbits && ob->matbits[i]) || do_mat_id_data_us)  && tmpmesh->mat[i]) { | 
					
						
							| 
									
										
										
										
											2015-11-09 19:47:10 +01:00
										 |  |  | 							id_us_plus(&tmpmesh->mat[i]->id); | 
					
						
							| 
									
										
											  
											
												Bake API - bpy.ops.object.bake()
New operator that can calls a bake function to the current render engine when available. This commit provides no feature for the users, but allows external engines to be accessed by the operator and be integrated with the baking api.
The API itself is simple. Blender sends a populated array of BakePixels to the renderer, and gets back an array of floats with the result.
The Blender Internal (and multires) system is still running independent, but we eventually will pipe it through the API as well. Cycles baking will come next as a separated commit
Python Operator:
----------------
The operator can be called with some arguments, or a user interface can be created for it. In that case the arguments can be ommited and the interface can expose the settings from bpy.context.scene.render.bake
bpy.ops.object.bake(type='COMBINED', filepath="", width=512, height=512, margin=16, use_selected_to_active=False, cage_extrusion=0, cage="", normal_space='TANGENT', normal_r='POS_X', normal_g='POS_Y', normal_b='POS_Z', save_mode='INTERNAL', use_clear=False, use_split_materials=False, use_automatic_name=False)
Note: external save mode is currently disabled.
Supported Features:
------------------
 * Margin - Baked result is extended this many pixels beyond the border of each UV "island," to soften seams in the texture.
 * Selected to Active - bake shading on the surface of selected object to the active object. The rays are cast from the lowpoly object inwards towards the highpoly object. If the highpoly object is not entirely involved by the lowpoly object, you can tweak the rays start point with Cage Extrusion. For even more control of the cage you can use a Cage object.
 * Cage Extrusion - distance to use for the inward ray cast when using selected to active
 * Custom Cage - object to use as cage (instead of the lowpoly object).
 * Normal swizzle - change the axis that gets mapped to RGB
 * Normal space - save as tangent or object normal spaces
Supported Passes:
-----------------
Any pass that is supported by Blender renderlayer system. Though it's up to the external engine to provide a valid enum with its supported passes. Normal passes get a special treatment since we post-process them to converted and "swizzled"
Development Notes for External Engines:
---------------------------------------
(read them in bake_api.c)
* For a complete implementation example look at the Cycles Bake commit (next).
Review: D421
Reviewed by: Campbell Barton, Brecht van Lommel, Sergey Sharybin, Thomas Dinge
Normal map pipeline "consulting" by Andy Davies (metalliandy)
Original design by Brecht van Lommel.
The entire commit history can be found on the branch: bake-cycles
											
										 
											2014-01-02 19:05:07 -02:00
										 |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 	} /* end copy materials */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (calc_tessface) { | 
					
						
							|  |  |  | 		/* cycles and exporters rely on this still */ | 
					
						
							|  |  |  | 		BKE_mesh_tessface_ensure(tmpmesh); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return tmpmesh; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-12 13:57:11 +05:00
										 |  |  | /* **** Depsgraph evaluation **** */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BKE_mesh_eval_geometry(EvaluationContext *UNUSED(eval_ctx), | 
					
						
							|  |  |  |                             Mesh *mesh) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (G.debug & G_DEBUG_DEPSGRAPH) { | 
					
						
							|  |  |  | 		printf("%s on %s\n", __func__, mesh->id.name); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (mesh->bb == NULL || (mesh->bb->flag & BOUNDBOX_DIRTY)) { | 
					
						
							|  |  |  | 		BKE_mesh_texspace_calc(mesh); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |