2006-08-28 01:12:36 +00:00
|
|
|
/*
|
2011-10-10 09:38:02 +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
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
* The Original Code is Copyright (C) 2006 Blender Foundation.
|
|
|
|
* All rights reserved.
|
|
|
|
* Implementation of CustomData.
|
|
|
|
*
|
|
|
|
* BKE_customdata.h contains the function prototypes for this file.
|
|
|
|
*/
|
2011-02-27 20:40:57 +00:00
|
|
|
|
2019-02-18 08:08:12 +11:00
|
|
|
/** \file
|
|
|
|
* \ingroup bke
|
2011-02-27 20:40:57 +00:00
|
|
|
*/
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2009-11-25 14:27:50 +00:00
|
|
|
#include "MEM_guardedalloc.h"
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2017-03-24 20:06:43 -03:00
|
|
|
#include "DNA_customdata_types.h"
|
2006-08-28 01:12:36 +00:00
|
|
|
#include "DNA_meshdata_types.h"
|
2009-11-25 14:27:50 +00:00
|
|
|
#include "DNA_ID.h"
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2011-05-09 04:06:48 +00:00
|
|
|
#include "BLI_utildefines.h"
|
2013-09-15 05:17:19 +00:00
|
|
|
#include "BLI_path_util.h"
|
2017-01-16 17:33:34 +01:00
|
|
|
#include "BLI_string.h"
|
|
|
|
#include "BLI_string_utils.h"
|
2017-03-26 04:08:16 -03:00
|
|
|
#include "BLI_math.h"
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
#include "BLI_math_color_blend.h"
|
2009-11-25 14:27:50 +00:00
|
|
|
#include "BLI_mempool.h"
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2015-08-16 17:32:01 +10:00
|
|
|
#include "BLT_translation.h"
|
2013-03-25 08:29:06 +00:00
|
|
|
|
2009-11-25 14:27:50 +00:00
|
|
|
#include "BKE_customdata.h"
|
2009-12-10 14:26:06 +00:00
|
|
|
#include "BKE_customdata_file.h"
|
2010-10-18 06:41:16 +00:00
|
|
|
#include "BKE_main.h"
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
#include "BKE_mesh_mapping.h"
|
|
|
|
#include "BKE_mesh_remap.h"
|
2010-11-04 16:00:28 +00:00
|
|
|
#include "BKE_multires.h"
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2009-06-23 05:35:49 +00:00
|
|
|
#include "bmesh.h"
|
|
|
|
|
2019-02-01 12:44:19 +11:00
|
|
|
#include "CLG_log.h"
|
|
|
|
|
2017-03-27 09:14:40 +11:00
|
|
|
/* only for customdata_data_transfer_interp_normal_normals */
|
|
|
|
#include "data_transfer_intern.h"
|
|
|
|
|
2006-08-28 01:12:36 +00:00
|
|
|
/* number of layers to add when growing a CustomData object */
|
|
|
|
#define CUSTOMDATA_GROW 5
|
|
|
|
|
2012-10-25 04:44:46 +00:00
|
|
|
/* ensure typemap size is ok */
|
2017-03-24 20:06:43 -03:00
|
|
|
BLI_STATIC_ASSERT(ARRAY_SIZE(((CustomData *)NULL)->typemap) == CD_NUMTYPES, "size mismatch");
|
2012-10-25 04:44:46 +00:00
|
|
|
|
2019-02-01 12:44:19 +11:00
|
|
|
static CLG_LogRef LOG = {"bke.customdata"};
|
2012-10-25 04:44:46 +00:00
|
|
|
|
Refactor CDData masks, to have one mask per mesh elem type.
We already have different storages for cddata of verts, edges etc.,
'simply' do the same for the mask flags we use all around Blender code
to request some data, or limit some operation to some layers, etc.
Reason we need this is that some cddata types (like Normals) are
actually shared between verts/polys/loops, and we don’t want to generate
clnors everytime we request vnors!
As a side note, this also does final fix to T59338, which was the
trigger for this patch (need to request computed loop normals for
another mesh than evaluated one).
Reviewers: brecht, campbellbarton, sergey
Differential Revision: https://developer.blender.org/D4407
2019-03-07 11:13:40 +01:00
|
|
|
/** Update mask_dst with layers defined in mask_src (equivalent to a bitwise OR). */
|
2019-04-17 06:17:24 +02:00
|
|
|
void CustomData_MeshMasks_update(CustomData_MeshMasks *mask_dst,
|
|
|
|
const CustomData_MeshMasks *mask_src)
|
Refactor CDData masks, to have one mask per mesh elem type.
We already have different storages for cddata of verts, edges etc.,
'simply' do the same for the mask flags we use all around Blender code
to request some data, or limit some operation to some layers, etc.
Reason we need this is that some cddata types (like Normals) are
actually shared between verts/polys/loops, and we don’t want to generate
clnors everytime we request vnors!
As a side note, this also does final fix to T59338, which was the
trigger for this patch (need to request computed loop normals for
another mesh than evaluated one).
Reviewers: brecht, campbellbarton, sergey
Differential Revision: https://developer.blender.org/D4407
2019-03-07 11:13:40 +01:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
mask_dst->vmask |= mask_src->vmask;
|
|
|
|
mask_dst->emask |= mask_src->emask;
|
|
|
|
mask_dst->fmask |= mask_src->fmask;
|
|
|
|
mask_dst->pmask |= mask_src->pmask;
|
|
|
|
mask_dst->lmask |= mask_src->lmask;
|
Refactor CDData masks, to have one mask per mesh elem type.
We already have different storages for cddata of verts, edges etc.,
'simply' do the same for the mask flags we use all around Blender code
to request some data, or limit some operation to some layers, etc.
Reason we need this is that some cddata types (like Normals) are
actually shared between verts/polys/loops, and we don’t want to generate
clnors everytime we request vnors!
As a side note, this also does final fix to T59338, which was the
trigger for this patch (need to request computed loop normals for
another mesh than evaluated one).
Reviewers: brecht, campbellbarton, sergey
Differential Revision: https://developer.blender.org/D4407
2019-03-07 11:13:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Return True if all layers set in \a mask_required are also set in \a mask_ref */
|
2019-04-17 06:17:24 +02:00
|
|
|
bool CustomData_MeshMasks_are_matching(const CustomData_MeshMasks *mask_ref,
|
|
|
|
const CustomData_MeshMasks *mask_required)
|
Refactor CDData masks, to have one mask per mesh elem type.
We already have different storages for cddata of verts, edges etc.,
'simply' do the same for the mask flags we use all around Blender code
to request some data, or limit some operation to some layers, etc.
Reason we need this is that some cddata types (like Normals) are
actually shared between verts/polys/loops, and we don’t want to generate
clnors everytime we request vnors!
As a side note, this also does final fix to T59338, which was the
trigger for this patch (need to request computed loop normals for
another mesh than evaluated one).
Reviewers: brecht, campbellbarton, sergey
Differential Revision: https://developer.blender.org/D4407
2019-03-07 11:13:40 +01:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
return (((mask_required->vmask & mask_ref->vmask) == mask_required->vmask) &&
|
|
|
|
((mask_required->emask & mask_ref->emask) == mask_required->emask) &&
|
|
|
|
((mask_required->fmask & mask_ref->fmask) == mask_required->fmask) &&
|
|
|
|
((mask_required->pmask & mask_ref->pmask) == mask_required->pmask) &&
|
|
|
|
((mask_required->lmask & mask_ref->lmask) == mask_required->lmask));
|
Refactor CDData masks, to have one mask per mesh elem type.
We already have different storages for cddata of verts, edges etc.,
'simply' do the same for the mask flags we use all around Blender code
to request some data, or limit some operation to some layers, etc.
Reason we need this is that some cddata types (like Normals) are
actually shared between verts/polys/loops, and we don’t want to generate
clnors everytime we request vnors!
As a side note, this also does final fix to T59338, which was the
trigger for this patch (need to request computed loop normals for
another mesh than evaluated one).
Reviewers: brecht, campbellbarton, sergey
Differential Revision: https://developer.blender.org/D4407
2019-03-07 11:13:40 +01:00
|
|
|
}
|
|
|
|
|
2006-08-28 01:12:36 +00:00
|
|
|
/********************* Layer type information **********************/
|
|
|
|
typedef struct LayerTypeInfo {
|
2019-04-17 06:17:24 +02:00
|
|
|
int size; /* the memory size of one element of this layer's data */
|
|
|
|
|
|
|
|
/** name of the struct used, for file writing */
|
|
|
|
const char *structname;
|
|
|
|
/** number of structs per element, for file writing */
|
|
|
|
int structnum;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* default layer name.
|
|
|
|
* note! when NULL this is a way to ensure there is only ever one item
|
|
|
|
* see: CustomData_layertype_is_singleton() */
|
|
|
|
const char *defaultname;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* a function to copy count elements of this layer's data
|
|
|
|
* (deep copy if appropriate)
|
|
|
|
* if NULL, memcpy is used
|
|
|
|
*/
|
|
|
|
cd_copy copy;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* a function to free any dynamically allocated components of this
|
|
|
|
* layer's data (note the data pointer itself should not be freed)
|
|
|
|
* size should be the size of one element of this layer's data (e.g.
|
|
|
|
* LayerTypeInfo.size)
|
|
|
|
*/
|
|
|
|
void (*free)(void *data, int count, int size);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* a function to interpolate between count source elements of this
|
|
|
|
* layer's data and store the result in dest
|
|
|
|
* if weights == NULL or sub_weights == NULL, they should default to 1
|
|
|
|
*
|
|
|
|
* weights gives the weight for each element in sources
|
|
|
|
* sub_weights gives the sub-element weights for each element in sources
|
|
|
|
* (there should be (sub element count)^2 weights per element)
|
|
|
|
* count gives the number of elements in sources
|
|
|
|
*
|
|
|
|
* \note in some cases \a dest pointer is in \a sources
|
|
|
|
* so all functions have to take this into account and delay
|
|
|
|
* applying changes while reading from sources.
|
|
|
|
* See bug [#32395] - Campbell.
|
|
|
|
*/
|
|
|
|
cd_interp interp;
|
|
|
|
|
|
|
|
/** a function to swap the data in corners of the element */
|
|
|
|
void (*swap)(void *data, const int *corner_indices);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* a function to set a layer's data to default values. if NULL, the
|
|
|
|
* default is assumed to be all zeros */
|
|
|
|
void (*set_default)(void *data, int count);
|
|
|
|
|
|
|
|
/** A function used by mesh validating code, must ensures passed item has valid data. */
|
|
|
|
cd_validate validate;
|
|
|
|
|
|
|
|
/** functions necessary for geometry collapse */
|
|
|
|
bool (*equal)(const void *data1, const void *data2);
|
|
|
|
void (*multiply)(void *data, float fac);
|
|
|
|
void (*initminmax)(void *min, void *max);
|
|
|
|
void (*add)(void *data1, const void *data2);
|
|
|
|
void (*dominmax)(const void *data1, void *min, void *max);
|
|
|
|
void (*copyvalue)(const void *source, void *dest, const int mixmode, const float mixfactor);
|
|
|
|
|
|
|
|
/** a function to read data from a cdf file */
|
|
|
|
int (*read)(CDataFile *cdf, void *data, int count);
|
|
|
|
|
|
|
|
/** a function to write data to a cdf file */
|
|
|
|
int (*write)(CDataFile *cdf, const void *data, int count);
|
|
|
|
|
|
|
|
/** a function to determine file size */
|
|
|
|
size_t (*filesize)(CDataFile *cdf, const void *data, int count);
|
|
|
|
|
2019-04-27 12:07:07 +10:00
|
|
|
/** a function to determine max allowed number of layers,
|
|
|
|
* should be NULL or return -1 if no limit */
|
2019-04-17 06:17:24 +02:00
|
|
|
int (*layers_max)(void);
|
2006-11-11 16:38:37 +00:00
|
|
|
} LayerTypeInfo;
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
static void layerCopy_mdeformvert(const void *source, void *dest, int count)
|
2006-08-28 01:12:36 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i, size = sizeof(MDeformVert);
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(dest, source, count * size);
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
MDeformVert *dvert = POINTER_OFFSET(dest, i * size);
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (dvert->totweight) {
|
|
|
|
MDeformWeight *dw = MEM_malloc_arrayN(
|
|
|
|
dvert->totweight, sizeof(*dw), "layerCopy_mdeformvert dw");
|
2008-04-15 18:07:04 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(dw, dvert->dw, dvert->totweight * sizeof(*dw));
|
|
|
|
dvert->dw = dw;
|
|
|
|
}
|
2019-04-22 09:39:35 +10:00
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
dvert->dw = NULL;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void layerFree_mdeformvert(void *data, int count, int size)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
MDeformVert *dvert = POINTER_OFFSET(data, i * size);
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (dvert->dw) {
|
|
|
|
MEM_freeN(dvert->dw);
|
|
|
|
dvert->dw = NULL;
|
|
|
|
dvert->totweight = 0;
|
|
|
|
}
|
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
2012-02-22 16:08:30 +00:00
|
|
|
/* copy just zeros in this case */
|
2019-04-17 06:17:24 +02:00
|
|
|
static void layerCopy_bmesh_elem_py_ptr(const void *UNUSED(source), void *dest, int count)
|
2012-02-22 16:08:30 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i, size = sizeof(void *);
|
2012-02-22 16:08:30 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
void **ptr = POINTER_OFFSET(dest, i * size);
|
|
|
|
*ptr = NULL;
|
|
|
|
}
|
2012-02-22 16:08:30 +00:00
|
|
|
}
|
|
|
|
|
2012-03-08 03:25:53 +00:00
|
|
|
#ifndef WITH_PYTHON
|
2013-03-16 20:49:46 +00:00
|
|
|
void bpy_bm_generic_invalidate(struct BPy_BMGeneric *UNUSED(self))
|
2012-03-08 03:25:53 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
/* dummy */
|
2012-03-08 03:25:53 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2012-02-22 16:08:30 +00:00
|
|
|
static void layerFree_bmesh_elem_py_ptr(void *data, int count, int size)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
void **ptr = POINTER_OFFSET(data, i * size);
|
|
|
|
if (*ptr) {
|
|
|
|
bpy_bm_generic_invalidate(*ptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void layerInterp_mdeformvert(const void **sources,
|
|
|
|
const float *weights,
|
|
|
|
const float *UNUSED(sub_weights),
|
|
|
|
int count,
|
|
|
|
void *dest)
|
|
|
|
{
|
|
|
|
/* a single linked list of MDeformWeight's
|
|
|
|
* use this to avoid double allocs (which LinkNode would do) */
|
|
|
|
struct MDeformWeight_Link {
|
|
|
|
struct MDeformWeight_Link *next;
|
|
|
|
MDeformWeight dw;
|
|
|
|
};
|
|
|
|
|
|
|
|
MDeformVert *dvert = dest;
|
|
|
|
struct MDeformWeight_Link *dest_dwlink = NULL;
|
|
|
|
struct MDeformWeight_Link *node;
|
|
|
|
int i, j, totweight;
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (count <= 0) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
/* build a list of unique def_nrs for dest */
|
|
|
|
totweight = 0;
|
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
const MDeformVert *source = sources[i];
|
|
|
|
float interp_weight = weights ? weights[i] : 1.0f;
|
|
|
|
|
|
|
|
for (j = 0; j < source->totweight; ++j) {
|
|
|
|
MDeformWeight *dw = &source->dw[j];
|
|
|
|
float weight = dw->weight * interp_weight;
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (weight == 0.0f) {
|
2019-04-17 06:17:24 +02:00
|
|
|
continue;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
for (node = dest_dwlink; node; node = node->next) {
|
|
|
|
MDeformWeight *tmp_dw = &node->dw;
|
|
|
|
|
|
|
|
if (tmp_dw->def_nr == dw->def_nr) {
|
|
|
|
tmp_dw->weight += weight;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if this def_nr is not in the list, add it */
|
|
|
|
if (!node) {
|
|
|
|
struct MDeformWeight_Link *tmp_dwlink = alloca(sizeof(*tmp_dwlink));
|
|
|
|
tmp_dwlink->dw.def_nr = dw->def_nr;
|
|
|
|
tmp_dwlink->dw.weight = weight;
|
|
|
|
|
|
|
|
/* inline linklist */
|
|
|
|
tmp_dwlink->next = dest_dwlink;
|
|
|
|
dest_dwlink = tmp_dwlink;
|
|
|
|
|
|
|
|
totweight++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* delay writing to the destination incase dest is in sources */
|
|
|
|
|
|
|
|
/* now we know how many unique deform weights there are, so realloc */
|
|
|
|
if (dvert->dw && (dvert->totweight == totweight)) {
|
|
|
|
/* pass (fastpath if we don't need to realloc) */
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (dvert->dw) {
|
|
|
|
MEM_freeN(dvert->dw);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (totweight) {
|
|
|
|
dvert->dw = MEM_malloc_arrayN(totweight, sizeof(*dvert->dw), __func__);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (totweight) {
|
|
|
|
dvert->totweight = totweight;
|
|
|
|
for (i = 0, node = dest_dwlink; node; node = node->next, i++) {
|
|
|
|
dvert->dw[i] = node->dw;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
memset(dvert, 0, sizeof(*dvert));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void layerInterp_normal(const void **sources,
|
|
|
|
const float *weights,
|
|
|
|
const float *UNUSED(sub_weights),
|
|
|
|
int count,
|
|
|
|
void *dest)
|
|
|
|
{
|
|
|
|
/* Note: This is linear interpolation, which is not optimal for vectors.
|
2019-04-27 12:07:07 +10:00
|
|
|
* Unfortunately, spherical interpolation of more than two values is hairy,
|
|
|
|
* so for now it will do... */
|
2019-04-17 06:17:24 +02:00
|
|
|
float no[3] = {0.0f};
|
|
|
|
|
|
|
|
while (count--) {
|
|
|
|
madd_v3_v3fl(no, (const float *)sources[count], weights[count]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Weighted sum of normalized vectors will **not** be normalized, even if weights are. */
|
|
|
|
normalize_v3_v3((float *)dest, no);
|
2015-02-05 14:38:59 +01:00
|
|
|
}
|
|
|
|
|
2018-12-03 16:19:08 +01:00
|
|
|
static bool layerValidate_normal(void *data, const uint totitems, const bool do_fixes)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
static const float no_default[3] = {0.0f, 0.0f, 1.0f}; /* Z-up default normal... */
|
|
|
|
float(*no)[3] = data;
|
|
|
|
bool has_errors = false;
|
|
|
|
|
|
|
|
for (int i = 0; i < totitems; i++, no++) {
|
|
|
|
if (!is_finite_v3((float *)no)) {
|
|
|
|
has_errors = true;
|
|
|
|
if (do_fixes) {
|
|
|
|
copy_v3_v3((float *)no, no_default);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (!compare_ff(len_squared_v3((float *)no), 1.0f, 1e-6f)) {
|
|
|
|
has_errors = true;
|
|
|
|
if (do_fixes) {
|
|
|
|
normalize_v3((float *)no);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return has_errors;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void layerCopyValue_normal(const void *source,
|
|
|
|
void *dest,
|
|
|
|
const int mixmode,
|
|
|
|
const float mixfactor)
|
|
|
|
{
|
|
|
|
const float *no_src = source;
|
|
|
|
float *no_dst = dest;
|
|
|
|
float no_tmp[3];
|
|
|
|
|
|
|
|
if (ELEM(mixmode,
|
|
|
|
CDT_MIX_NOMIX,
|
|
|
|
CDT_MIX_REPLACE_ABOVE_THRESHOLD,
|
|
|
|
CDT_MIX_REPLACE_BELOW_THRESHOLD)) {
|
|
|
|
/* Above/below threshold modes are not supported here, fallback to nomix (just in case). */
|
|
|
|
copy_v3_v3(no_dst, no_src);
|
|
|
|
}
|
|
|
|
else { /* Modes that support 'real' mix factor. */
|
|
|
|
/* Since we normalize in the end, MIX and ADD are the same op here. */
|
|
|
|
if (ELEM(mixmode, CDT_MIX_MIX, CDT_MIX_ADD)) {
|
|
|
|
add_v3_v3v3(no_tmp, no_dst, no_src);
|
|
|
|
normalize_v3(no_tmp);
|
|
|
|
}
|
|
|
|
else if (mixmode == CDT_MIX_SUB) {
|
|
|
|
sub_v3_v3v3(no_tmp, no_dst, no_src);
|
|
|
|
normalize_v3(no_tmp);
|
|
|
|
}
|
|
|
|
else if (mixmode == CDT_MIX_MUL) {
|
|
|
|
mul_v3_v3v3(no_tmp, no_dst, no_src);
|
|
|
|
normalize_v3(no_tmp);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
copy_v3_v3(no_tmp, no_src);
|
|
|
|
}
|
|
|
|
interp_v3_v3v3_slerp_safe(no_dst, no_dst, no_tmp, mixfactor);
|
|
|
|
}
|
2015-02-05 14:38:59 +01:00
|
|
|
}
|
|
|
|
|
2006-12-12 21:29:09 +00:00
|
|
|
static void layerCopy_tface(const void *source, void *dest, int count)
|
2006-11-11 16:38:37 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const MTFace *source_tf = (const MTFace *)source;
|
|
|
|
MTFace *dest_tf = (MTFace *)dest;
|
|
|
|
int i;
|
2006-11-11 16:38:37 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < count; ++i) {
|
2019-04-17 06:17:24 +02:00
|
|
|
dest_tf[i] = source_tf[i];
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2006-11-11 16:38:37 +00:00
|
|
|
}
|
|
|
|
|
2015-02-23 13:51:55 +11:00
|
|
|
static void layerInterp_tface(
|
2019-04-17 06:17:24 +02:00
|
|
|
const void **sources, const float *weights, const float *sub_weights, int count, void *dest)
|
|
|
|
{
|
|
|
|
MTFace *tf = dest;
|
|
|
|
int i, j, k;
|
|
|
|
float uv[4][2] = {{0.0f}};
|
|
|
|
const float *sub_weight;
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (count <= 0) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
sub_weight = sub_weights;
|
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
float weight = weights ? weights[i] : 1;
|
|
|
|
const MTFace *src = sources[i];
|
|
|
|
|
|
|
|
for (j = 0; j < 4; ++j) {
|
|
|
|
if (sub_weights) {
|
|
|
|
for (k = 0; k < 4; ++k, ++sub_weight) {
|
|
|
|
madd_v2_v2fl(uv[j], src->uv[k], (*sub_weight) * weight);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
madd_v2_v2fl(uv[j], src->uv[j], weight);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* delay writing to the destination incase dest is in sources */
|
|
|
|
*tf = *(MTFace *)(*sources);
|
|
|
|
memcpy(tf->uv, uv, sizeof(tf->uv));
|
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
|
|
|
}
|
|
|
|
|
2010-04-10 22:12:10 +00:00
|
|
|
static void layerSwap_tface(void *data, const int *corner_indices)
|
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
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
MTFace *tf = data;
|
|
|
|
float uv[4][2];
|
|
|
|
int j;
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (j = 0; j < 4; ++j) {
|
|
|
|
const int source_index = corner_indices[j];
|
|
|
|
copy_v2_v2(uv[j], tf->uv[source_index]);
|
|
|
|
}
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(tf->uv, uv, sizeof(tf->uv));
|
2006-08-28 01:12:36 +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
|
|
|
static void layerDefault_tface(void *data, int count)
|
2006-11-11 16:38:37 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
static MTFace default_tf = {{{0, 0}, {1, 0}, {1, 1}, {0, 1}}};
|
|
|
|
MTFace *tf = (MTFace *)data;
|
|
|
|
int i;
|
2006-11-11 16:38:37 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < count; i++) {
|
2019-04-17 06:17:24 +02:00
|
|
|
tf[i] = default_tf;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2006-11-11 16:38:37 +00:00
|
|
|
}
|
|
|
|
|
2014-10-21 17:01:56 +02:00
|
|
|
static int layerMaxNum_tface(void)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
return MAX_MTFACE;
|
2014-10-21 17:01:56 +02:00
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
static void layerCopy_propFloat(const void *source, void *dest, int count)
|
2011-04-16 13:00:41 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(dest, source, sizeof(MFloatProperty) * count);
|
2011-04-16 13:00:41 +00:00
|
|
|
}
|
|
|
|
|
2018-12-03 16:19:08 +01:00
|
|
|
static bool layerValidate_propFloat(void *data, const uint totitems, const bool do_fixes)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
MFloatProperty *fp = data;
|
|
|
|
bool has_errors = false;
|
2018-12-03 16:19:08 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (int i = 0; i < totitems; i++, fp++) {
|
|
|
|
if (!isfinite(fp->f)) {
|
|
|
|
if (do_fixes) {
|
|
|
|
fp->f = 0.0f;
|
|
|
|
}
|
|
|
|
has_errors = true;
|
|
|
|
}
|
|
|
|
}
|
2018-12-03 16:19:08 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return has_errors;
|
2018-12-03 16:19:08 +01:00
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
static void layerCopy_propInt(const void *source, void *dest, int count)
|
2011-04-16 13:00:41 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(dest, source, sizeof(MIntProperty) * count);
|
2011-04-16 13:00:41 +00:00
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
static void layerCopy_propString(const void *source, void *dest, int count)
|
2011-04-16 13:00:41 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(dest, source, sizeof(MStringProperty) * count);
|
2011-04-16 13:00:41 +00:00
|
|
|
}
|
|
|
|
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
static void layerCopy_origspace_face(const void *source, void *dest, int count)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const OrigSpaceFace *source_tf = (const OrigSpaceFace *)source;
|
|
|
|
OrigSpaceFace *dest_tf = (OrigSpaceFace *)dest;
|
|
|
|
int i;
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < count; ++i) {
|
2019-04-17 06:17:24 +02:00
|
|
|
dest_tf[i] = source_tf[i];
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
}
|
|
|
|
|
2015-02-23 13:51:55 +11:00
|
|
|
static void layerInterp_origspace_face(
|
2019-04-17 06:17:24 +02:00
|
|
|
const void **sources, const float *weights, const float *sub_weights, int count, void *dest)
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
OrigSpaceFace *osf = dest;
|
|
|
|
int i, j, k;
|
|
|
|
float uv[4][2] = {{0.0f}};
|
|
|
|
const float *sub_weight;
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (count <= 0) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
sub_weight = sub_weights;
|
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
float weight = weights ? weights[i] : 1;
|
|
|
|
const OrigSpaceFace *src = sources[i];
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (j = 0; j < 4; ++j) {
|
|
|
|
if (sub_weights) {
|
|
|
|
for (k = 0; k < 4; ++k, ++sub_weight) {
|
|
|
|
madd_v2_v2fl(uv[j], src->uv[k], (*sub_weight) * weight);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
madd_v2_v2fl(uv[j], src->uv[j], weight);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* delay writing to the destination in case dest is in sources */
|
|
|
|
memcpy(osf->uv, uv, sizeof(osf->uv));
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
}
|
|
|
|
|
2010-04-10 22:12:10 +00:00
|
|
|
static void layerSwap_origspace_face(void *data, const int *corner_indices)
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
OrigSpaceFace *osf = data;
|
|
|
|
float uv[4][2];
|
|
|
|
int j;
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (j = 0; j < 4; ++j) {
|
|
|
|
copy_v2_v2(uv[j], osf->uv[corner_indices[j]]);
|
|
|
|
}
|
|
|
|
memcpy(osf->uv, uv, sizeof(osf->uv));
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void layerDefault_origspace_face(void *data, int count)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
static OrigSpaceFace default_osf = {{{0, 0}, {1, 0}, {1, 1}, {0, 1}}};
|
|
|
|
OrigSpaceFace *osf = (OrigSpaceFace *)data;
|
|
|
|
int i;
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < count; i++) {
|
2019-04-17 06:17:24 +02:00
|
|
|
osf[i] = default_osf;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
}
|
2009-01-06 18:59:03 +00:00
|
|
|
|
2010-04-17 15:47:00 +00:00
|
|
|
static void layerSwap_mdisps(void *data, const int *ci)
|
2009-01-06 18:59:03 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
MDisps *s = data;
|
|
|
|
float(*d)[3] = NULL;
|
|
|
|
int corners, cornersize, S;
|
2009-01-06 18:59:03 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (s->disps) {
|
|
|
|
int nverts = (ci[1] == 3) ? 4 : 3; /* silly way to know vertex count of face */
|
|
|
|
corners = multires_mdisp_corners(s);
|
|
|
|
cornersize = s->totdisp / corners;
|
2009-01-06 18:59:03 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (corners != nverts) {
|
|
|
|
/* happens when face changed vertex count in edit mode
|
|
|
|
* if it happened, just forgot displacement */
|
2009-01-06 18:59:03 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
MEM_freeN(s->disps);
|
|
|
|
s->totdisp = (s->totdisp / corners) * nverts;
|
|
|
|
s->disps = MEM_calloc_arrayN(s->totdisp, sizeof(float) * 3, "mdisp swap");
|
|
|
|
return;
|
|
|
|
}
|
2010-10-16 20:43:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
d = MEM_calloc_arrayN(s->totdisp, 3 * sizeof(float), "mdisps swap");
|
2009-01-06 18:59:03 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (S = 0; S < corners; S++) {
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(d + cornersize * S, s->disps + cornersize * ci[S], cornersize * 3 * sizeof(float));
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
MEM_freeN(s->disps);
|
|
|
|
s->disps = d;
|
|
|
|
}
|
2009-01-06 18:59:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void layerCopy_mdisps(const void *source, void *dest, int count)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
|
|
|
const MDisps *s = source;
|
|
|
|
MDisps *d = dest;
|
2009-01-06 18:59:03 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
if (s[i].disps) {
|
|
|
|
d[i].disps = MEM_dupallocN(s[i].disps);
|
|
|
|
d[i].hidden = MEM_dupallocN(s[i].hidden);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
d[i].disps = NULL;
|
|
|
|
d[i].hidden = NULL;
|
|
|
|
}
|
2013-11-26 17:09:15 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* still copy even if not in memory, displacement can be external */
|
|
|
|
d[i].totdisp = s[i].totdisp;
|
|
|
|
d[i].level = s[i].level;
|
|
|
|
}
|
2009-01-06 18:59:03 +00:00
|
|
|
}
|
|
|
|
|
2010-10-17 06:38:56 +00:00
|
|
|
static void layerFree_mdisps(void *data, int count, int UNUSED(size))
|
2009-01-06 18:59:03 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
|
|
|
MDisps *d = data;
|
2009-01-06 18:59:03 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < count; ++i) {
|
2019-04-22 09:39:35 +10:00
|
|
|
if (d[i].disps) {
|
2019-04-17 06:17:24 +02:00
|
|
|
MEM_freeN(d[i].disps);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (d[i].hidden) {
|
2019-04-17 06:17:24 +02:00
|
|
|
MEM_freeN(d[i].hidden);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
d[i].disps = NULL;
|
|
|
|
d[i].hidden = NULL;
|
|
|
|
d[i].totdisp = 0;
|
|
|
|
d[i].level = 0;
|
|
|
|
}
|
2009-01-06 18:59:03 +00:00
|
|
|
}
|
|
|
|
|
2009-12-10 14:26:06 +00:00
|
|
|
static int layerRead_mdisps(CDataFile *cdf, void *data, int count)
|
2009-11-25 14:27:50 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
MDisps *d = data;
|
|
|
|
int i;
|
2009-11-25 14:27:50 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < count; ++i) {
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!d[i].disps) {
|
2019-04-17 06:17:24 +02:00
|
|
|
d[i].disps = MEM_calloc_arrayN(d[i].totdisp, 3 * sizeof(float), "mdisps read");
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2009-11-25 14:27:50 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (!cdf_read_data(cdf, d[i].totdisp * 3 * sizeof(float), d[i].disps)) {
|
|
|
|
CLOG_ERROR(&LOG, "failed to read multires displacement %d/%d %d", i, count, d[i].totdisp);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2009-11-25 14:27:50 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return 1;
|
2009-11-25 14:27:50 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
static int layerWrite_mdisps(CDataFile *cdf, const void *data, int count)
|
2009-11-25 14:27:50 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const MDisps *d = data;
|
|
|
|
int i;
|
2009-11-25 14:27:50 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
if (!cdf_write_data(cdf, d[i].totdisp * 3 * sizeof(float), d[i].disps)) {
|
|
|
|
CLOG_ERROR(&LOG, "failed to write multires displacement %d/%d %d", i, count, d[i].totdisp);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2009-11-25 14:27:50 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return 1;
|
2009-11-25 14:27:50 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
static size_t layerFilesize_mdisps(CDataFile *UNUSED(cdf), const void *data, int count)
|
2009-12-10 14:26:06 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const MDisps *d = data;
|
|
|
|
size_t size = 0;
|
|
|
|
int i;
|
2009-12-10 14:26:06 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < count; ++i) {
|
2019-04-17 06:17:24 +02:00
|
|
|
size += d[i].totdisp * 3 * sizeof(float);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2009-12-10 14:26:06 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return size;
|
2009-12-10 14:26:06 +00:00
|
|
|
}
|
|
|
|
|
2012-05-10 20:33:24 +00:00
|
|
|
static void layerCopy_grid_paint_mask(const void *source, void *dest, int count)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
|
|
|
const GridPaintMask *s = source;
|
|
|
|
GridPaintMask *d = dest;
|
2012-05-10 20:33:24 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
if (s[i].data) {
|
|
|
|
d[i].data = MEM_dupallocN(s[i].data);
|
|
|
|
d[i].level = s[i].level;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
d[i].data = NULL;
|
|
|
|
d[i].level = 0;
|
|
|
|
}
|
|
|
|
}
|
2012-05-10 20:33:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void layerFree_grid_paint_mask(void *data, int count, int UNUSED(size))
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
|
|
|
GridPaintMask *gpm = data;
|
2012-05-10 20:33:24 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < count; ++i) {
|
2019-04-22 09:39:35 +10:00
|
|
|
if (gpm[i].data) {
|
2019-04-17 06:17:24 +02:00
|
|
|
MEM_freeN(gpm[i].data);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
gpm[i].data = NULL;
|
|
|
|
gpm[i].level = 0;
|
|
|
|
}
|
2012-05-10 20:33:24 +00:00
|
|
|
}
|
|
|
|
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
/* --------- */
|
2019-04-17 06:17:24 +02:00
|
|
|
static void layerCopyValue_mloopcol(const void *source,
|
|
|
|
void *dest,
|
|
|
|
const int mixmode,
|
|
|
|
const float mixfactor)
|
|
|
|
{
|
|
|
|
const MLoopCol *m1 = source;
|
|
|
|
MLoopCol *m2 = dest;
|
|
|
|
unsigned char tmp_col[4];
|
|
|
|
|
|
|
|
if (ELEM(mixmode,
|
|
|
|
CDT_MIX_NOMIX,
|
|
|
|
CDT_MIX_REPLACE_ABOVE_THRESHOLD,
|
|
|
|
CDT_MIX_REPLACE_BELOW_THRESHOLD)) {
|
|
|
|
/* Modes that do a full copy or nothing. */
|
|
|
|
if (ELEM(mixmode, CDT_MIX_REPLACE_ABOVE_THRESHOLD, CDT_MIX_REPLACE_BELOW_THRESHOLD)) {
|
|
|
|
/* TODO: Check for a real valid way to get 'factor' value of our dest color? */
|
|
|
|
const float f = ((float)m2->r + (float)m2->g + (float)m2->b) / 3.0f;
|
|
|
|
if (mixmode == CDT_MIX_REPLACE_ABOVE_THRESHOLD && f < mixfactor) {
|
|
|
|
return; /* Do Nothing! */
|
|
|
|
}
|
|
|
|
else if (mixmode == CDT_MIX_REPLACE_BELOW_THRESHOLD && f > mixfactor) {
|
|
|
|
return; /* Do Nothing! */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m2->r = m1->r;
|
|
|
|
m2->g = m1->g;
|
|
|
|
m2->b = m1->b;
|
|
|
|
}
|
|
|
|
else { /* Modes that support 'real' mix factor. */
|
|
|
|
unsigned char src[4] = {m1->r, m1->g, m1->b, m1->a};
|
|
|
|
unsigned char dst[4] = {m2->r, m2->g, m2->b, m2->a};
|
|
|
|
|
|
|
|
if (mixmode == CDT_MIX_MIX) {
|
|
|
|
blend_color_mix_byte(tmp_col, dst, src);
|
|
|
|
}
|
|
|
|
else if (mixmode == CDT_MIX_ADD) {
|
|
|
|
blend_color_add_byte(tmp_col, dst, src);
|
|
|
|
}
|
|
|
|
else if (mixmode == CDT_MIX_SUB) {
|
|
|
|
blend_color_sub_byte(tmp_col, dst, src);
|
|
|
|
}
|
|
|
|
else if (mixmode == CDT_MIX_MUL) {
|
|
|
|
blend_color_mul_byte(tmp_col, dst, src);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
memcpy(tmp_col, src, sizeof(tmp_col));
|
|
|
|
}
|
|
|
|
blend_color_interpolate_byte(dst, dst, tmp_col, mixfactor);
|
|
|
|
|
|
|
|
m2->r = (char)dst[0];
|
|
|
|
m2->g = (char)dst[1];
|
|
|
|
m2->b = (char)dst[2];
|
|
|
|
}
|
|
|
|
m2->a = m1->a;
|
2009-08-12 03:51:28 +00:00
|
|
|
}
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
static bool layerEqual_mloopcol(const void *data1, const void *data2)
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const MLoopCol *m1 = data1, *m2 = data2;
|
|
|
|
float r, g, b, a;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
r = m1->r - m2->r;
|
|
|
|
g = m1->g - m2->g;
|
|
|
|
b = m1->b - m2->b;
|
|
|
|
a = m1->a - m2->a;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return r * r + g * g + b * b + a * a < 0.001f;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void layerMultiply_mloopcol(void *data, float fac)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
MLoopCol *m = data;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
m->r = (float)m->r * fac;
|
|
|
|
m->g = (float)m->g * fac;
|
|
|
|
m->b = (float)m->b * fac;
|
|
|
|
m->a = (float)m->a * fac;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
static void layerAdd_mloopcol(void *data1, const void *data2)
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
MLoopCol *m = data1;
|
|
|
|
const MLoopCol *m2 = data2;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
m->r += m2->r;
|
|
|
|
m->g += m2->g;
|
|
|
|
m->b += m2->b;
|
|
|
|
m->a += m2->a;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
static void layerDoMinMax_mloopcol(const void *data, void *vmin, void *vmax)
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const MLoopCol *m = data;
|
|
|
|
MLoopCol *min = vmin, *max = vmax;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (m->r < min->r) {
|
2019-04-17 06:17:24 +02:00
|
|
|
min->r = m->r;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (m->g < min->g) {
|
2019-04-17 06:17:24 +02:00
|
|
|
min->g = m->g;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (m->b < min->b) {
|
2019-04-17 06:17:24 +02:00
|
|
|
min->b = m->b;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (m->a < min->a) {
|
2019-04-17 06:17:24 +02:00
|
|
|
min->a = m->a;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (m->r > max->r) {
|
2019-04-17 06:17:24 +02:00
|
|
|
max->r = m->r;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (m->g > max->g) {
|
2019-04-17 06:17:24 +02:00
|
|
|
max->g = m->g;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (m->b > max->b) {
|
2019-04-17 06:17:24 +02:00
|
|
|
max->b = m->b;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (m->a > max->a) {
|
2019-04-17 06:17:24 +02:00
|
|
|
max->a = m->a;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void layerInitMinMax_mloopcol(void *vmin, void *vmax)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
MLoopCol *min = vmin, *max = vmax;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
min->r = 255;
|
|
|
|
min->g = 255;
|
|
|
|
min->b = 255;
|
|
|
|
min->a = 255;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
max->r = 0;
|
|
|
|
max->g = 0;
|
|
|
|
max->b = 0;
|
|
|
|
max->a = 0;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
2008-07-08 02:22:37 +00:00
|
|
|
static void layerDefault_mloopcol(void *data, int count)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
MLoopCol default_mloopcol = {255, 255, 255, 255};
|
|
|
|
MLoopCol *mlcol = (MLoopCol *)data;
|
|
|
|
int i;
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < count; i++) {
|
2019-04-17 06:17:24 +02:00
|
|
|
mlcol[i] = default_mloopcol;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2008-07-08 02:22:37 +00:00
|
|
|
}
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
|
2015-02-23 13:51:55 +11:00
|
|
|
static void layerInterp_mloopcol(
|
2019-04-17 06:17:24 +02:00
|
|
|
const void **sources, const float *weights, const float *sub_weights, int count, void *dest)
|
|
|
|
{
|
|
|
|
MLoopCol *mc = dest;
|
|
|
|
struct {
|
|
|
|
float a;
|
|
|
|
float r;
|
|
|
|
float g;
|
|
|
|
float b;
|
|
|
|
} col = {0};
|
|
|
|
|
|
|
|
const float *sub_weight = sub_weights;
|
|
|
|
for (int i = 0; i < count; ++i) {
|
|
|
|
float weight = weights ? weights[i] : 1;
|
|
|
|
const MLoopCol *src = sources[i];
|
|
|
|
if (sub_weights) {
|
|
|
|
col.r += src->r * (*sub_weight) * weight;
|
|
|
|
col.g += src->g * (*sub_weight) * weight;
|
|
|
|
col.b += src->b * (*sub_weight) * weight;
|
|
|
|
col.a += src->a * (*sub_weight) * weight;
|
|
|
|
sub_weight++;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
col.r += src->r * weight;
|
|
|
|
col.g += src->g * weight;
|
|
|
|
col.b += src->b * weight;
|
|
|
|
col.a += src->a * weight;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Subdivide smooth or fractal can cause problems without clamping
|
|
|
|
* although weights should also not cause this situation */
|
|
|
|
|
|
|
|
/* also delay writing to the destination incase dest is in sources */
|
|
|
|
mc->r = round_fl_to_uchar_clamp(col.r);
|
|
|
|
mc->g = round_fl_to_uchar_clamp(col.g);
|
|
|
|
mc->b = round_fl_to_uchar_clamp(col.b);
|
|
|
|
mc->a = round_fl_to_uchar_clamp(col.a);
|
2008-07-08 02:22:37 +00:00
|
|
|
}
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2014-10-21 17:01:56 +02:00
|
|
|
static int layerMaxNum_mloopcol(void)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
return MAX_MCOL;
|
2014-10-21 17:01:56 +02:00
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
static void layerCopyValue_mloopuv(const void *source,
|
|
|
|
void *dest,
|
|
|
|
const int mixmode,
|
|
|
|
const float mixfactor)
|
2009-08-12 03:51:28 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const MLoopUV *luv1 = source;
|
|
|
|
MLoopUV *luv2 = dest;
|
2011-12-29 11:18:12 +00:00
|
|
|
|
2019-04-27 12:07:07 +10:00
|
|
|
/* We only support a limited subset of advanced mixing here -
|
|
|
|
* namely the mixfactor interpolation. */
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (mixmode == CDT_MIX_NOMIX) {
|
|
|
|
copy_v2_v2(luv2->uv, luv1->uv);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
interp_v2_v2v2(luv2->uv, luv2->uv, luv1->uv, mixfactor);
|
|
|
|
}
|
2009-08-12 03:51:28 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
static bool layerEqual_mloopuv(const void *data1, const void *data2)
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const MLoopUV *luv1 = data1, *luv2 = data2;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return len_squared_v2v2(luv1->uv, luv2->uv) < 0.00001f;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void layerMultiply_mloopuv(void *data, float fac)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
MLoopUV *luv = data;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
mul_v2_fl(luv->uv, fac);
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void layerInitMinMax_mloopuv(void *vmin, void *vmax)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
MLoopUV *min = vmin, *max = vmax;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
INIT_MINMAX2(min->uv, max->uv);
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
static void layerDoMinMax_mloopuv(const void *data, void *vmin, void *vmax)
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const MLoopUV *luv = data;
|
|
|
|
MLoopUV *min = vmin, *max = vmax;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
minmax_v2v2_v2(min->uv, max->uv, luv->uv);
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
static void layerAdd_mloopuv(void *data1, const void *data2)
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
MLoopUV *l1 = data1;
|
|
|
|
const MLoopUV *l2 = data2;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
add_v2_v2(l1->uv, l2->uv);
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
2015-02-23 13:51:55 +11:00
|
|
|
static void layerInterp_mloopuv(
|
2019-04-17 06:17:24 +02:00
|
|
|
const void **sources, const float *weights, const float *sub_weights, int count, void *dest)
|
|
|
|
{
|
|
|
|
float uv[2];
|
|
|
|
int flag = 0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
zero_v2(uv);
|
|
|
|
|
|
|
|
if (sub_weights) {
|
|
|
|
const float *sub_weight = sub_weights;
|
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
float weight = (weights ? weights[i] : 1.0f) * (*sub_weight);
|
|
|
|
const MLoopUV *src = sources[i];
|
|
|
|
madd_v2_v2fl(uv, src->uv, weight);
|
|
|
|
if (weight > 0.0f) {
|
|
|
|
flag |= src->flag;
|
|
|
|
}
|
|
|
|
sub_weight++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
float weight = weights ? weights[i] : 1;
|
|
|
|
const MLoopUV *src = sources[i];
|
|
|
|
madd_v2_v2fl(uv, src->uv, weight);
|
|
|
|
if (weight > 0.0f) {
|
|
|
|
flag |= src->flag;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* delay writing to the destination incase dest is in sources */
|
|
|
|
copy_v2_v2(((MLoopUV *)dest)->uv, uv);
|
|
|
|
((MLoopUV *)dest)->flag = flag;
|
2008-07-08 02:22:37 +00:00
|
|
|
}
|
Particles
=========
Merge of the famous particle patch by Janne Karhu, a full rewrite
of the Blender particle system. This includes:
- Emitter, Hair and Reactor particle types.
- Newtonian, Keyed and Boids physics.
- Various particle visualisation and rendering types.
- Vertex group and texture control for various properties.
- Interpolated child particles from parents.
- Hair editing with combing, growing, cutting, .. .
- Explode modifier.
- Harmonic, Magnetic fields, and multiple falloff types.
.. and lots of other things, some more info is here:
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite
http://wiki.blender.org/index.php/BlenderDev/Particles_Rewrite_Doc
The new particle system cannot be backwards compatible. Old particle
systems are being converted to the new system, but will require
tweaking to get them looking the same as before.
Point Cache
===========
The new system to replace manual baking, based on automatic caching
on disk. This is currently used by softbodies and the particle system.
See the Cache API section on:
http://wiki.blender.org/index.php/BlenderDev/PhysicsSprint
Documentation
=============
These new features still need good docs for the release logs, help
for this is appreciated.
2007-11-26 22:09:57 +00:00
|
|
|
|
2018-12-03 16:19:08 +01:00
|
|
|
static bool layerValidate_mloopuv(void *data, const uint totitems, const bool do_fixes)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
MLoopUV *uv = data;
|
|
|
|
bool has_errors = false;
|
2018-12-03 16:19:08 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (int i = 0; i < totitems; i++, uv++) {
|
|
|
|
if (!is_finite_v2(uv->uv)) {
|
|
|
|
if (do_fixes) {
|
|
|
|
zero_v2(uv->uv);
|
|
|
|
}
|
|
|
|
has_errors = true;
|
|
|
|
}
|
|
|
|
}
|
2018-12-03 16:19:08 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return has_errors;
|
2018-12-03 16:19:08 +01:00
|
|
|
}
|
|
|
|
|
2012-02-05 11:30:26 +00:00
|
|
|
/* origspace is almost exact copy of mloopuv's, keep in sync */
|
2019-04-17 06:17:24 +02:00
|
|
|
static void layerCopyValue_mloop_origspace(const void *source,
|
|
|
|
void *dest,
|
|
|
|
const int UNUSED(mixmode),
|
|
|
|
const float UNUSED(mixfactor))
|
2012-02-05 11:30:26 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const OrigSpaceLoop *luv1 = source;
|
|
|
|
OrigSpaceLoop *luv2 = dest;
|
2012-02-05 11:30:26 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
copy_v2_v2(luv2->uv, luv1->uv);
|
2012-02-05 11:30:26 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
static bool layerEqual_mloop_origspace(const void *data1, const void *data2)
|
2012-02-05 11:30:26 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const OrigSpaceLoop *luv1 = data1, *luv2 = data2;
|
2012-02-05 11:30:26 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return len_squared_v2v2(luv1->uv, luv2->uv) < 0.00001f;
|
2012-02-05 11:30:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void layerMultiply_mloop_origspace(void *data, float fac)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
OrigSpaceLoop *luv = data;
|
2012-02-05 11:30:26 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
mul_v2_fl(luv->uv, fac);
|
2012-02-05 11:30:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void layerInitMinMax_mloop_origspace(void *vmin, void *vmax)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
OrigSpaceLoop *min = vmin, *max = vmax;
|
2012-02-05 11:30:26 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
INIT_MINMAX2(min->uv, max->uv);
|
2012-02-05 11:30:26 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
static void layerDoMinMax_mloop_origspace(const void *data, void *vmin, void *vmax)
|
2012-02-05 11:30:26 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const OrigSpaceLoop *luv = data;
|
|
|
|
OrigSpaceLoop *min = vmin, *max = vmax;
|
2012-02-05 11:30:26 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
minmax_v2v2_v2(min->uv, max->uv, luv->uv);
|
2012-02-05 11:30:26 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
static void layerAdd_mloop_origspace(void *data1, const void *data2)
|
2012-02-05 11:30:26 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
OrigSpaceLoop *l1 = data1;
|
|
|
|
const OrigSpaceLoop *l2 = data2;
|
2012-02-05 11:30:26 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
add_v2_v2(l1->uv, l2->uv);
|
2012-02-05 11:30:26 +00:00
|
|
|
}
|
|
|
|
|
2015-02-23 13:51:55 +11:00
|
|
|
static void layerInterp_mloop_origspace(
|
2019-04-17 06:17:24 +02:00
|
|
|
const void **sources, const float *weights, const float *sub_weights, int count, void *dest)
|
|
|
|
{
|
|
|
|
float uv[2];
|
|
|
|
int i;
|
|
|
|
|
|
|
|
zero_v2(uv);
|
|
|
|
|
|
|
|
if (sub_weights) {
|
|
|
|
const float *sub_weight = sub_weights;
|
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
float weight = weights ? weights[i] : 1.0f;
|
|
|
|
const OrigSpaceLoop *src = sources[i];
|
|
|
|
madd_v2_v2fl(uv, src->uv, (*sub_weight) * weight);
|
|
|
|
sub_weight++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
float weight = weights ? weights[i] : 1.0f;
|
|
|
|
const OrigSpaceLoop *src = sources[i];
|
|
|
|
madd_v2_v2fl(uv, src->uv, weight);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* delay writing to the destination incase dest is in sources */
|
|
|
|
copy_v2_v2(((OrigSpaceLoop *)dest)->uv, uv);
|
2012-02-05 11:30:26 +00:00
|
|
|
}
|
|
|
|
/* --- end copy */
|
|
|
|
|
2015-02-23 13:51:55 +11:00
|
|
|
static void layerInterp_mcol(
|
2019-04-17 06:17:24 +02:00
|
|
|
const void **sources, const float *weights, const float *sub_weights, int count, void *dest)
|
|
|
|
{
|
|
|
|
MCol *mc = dest;
|
|
|
|
int i, j, k;
|
|
|
|
struct {
|
|
|
|
float a;
|
|
|
|
float r;
|
|
|
|
float g;
|
|
|
|
float b;
|
|
|
|
} col[4] = {{0.0f}};
|
|
|
|
|
|
|
|
const float *sub_weight;
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (count <= 0) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
sub_weight = sub_weights;
|
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
float weight = weights ? weights[i] : 1;
|
|
|
|
|
|
|
|
for (j = 0; j < 4; ++j) {
|
|
|
|
if (sub_weights) {
|
|
|
|
const MCol *src = sources[i];
|
|
|
|
for (k = 0; k < 4; ++k, ++sub_weight, ++src) {
|
|
|
|
const float w = (*sub_weight) * weight;
|
|
|
|
col[j].a += src->a * w;
|
|
|
|
col[j].r += src->r * w;
|
|
|
|
col[j].g += src->g * w;
|
|
|
|
col[j].b += src->b * w;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
const MCol *src = sources[i];
|
|
|
|
col[j].a += src[j].a * weight;
|
|
|
|
col[j].r += src[j].r * weight;
|
|
|
|
col[j].g += src[j].g * weight;
|
|
|
|
col[j].b += src[j].b * weight;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* delay writing to the destination incase dest is in sources */
|
|
|
|
for (j = 0; j < 4; ++j) {
|
|
|
|
|
|
|
|
/* Subdivide smooth or fractal can cause problems without clamping
|
|
|
|
* although weights should also not cause this situation */
|
|
|
|
mc[j].a = round_fl_to_uchar_clamp(col[j].a);
|
|
|
|
mc[j].r = round_fl_to_uchar_clamp(col[j].r);
|
|
|
|
mc[j].g = round_fl_to_uchar_clamp(col[j].g);
|
|
|
|
mc[j].b = round_fl_to_uchar_clamp(col[j].b);
|
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
2010-04-10 22:12:10 +00:00
|
|
|
static void layerSwap_mcol(void *data, const int *corner_indices)
|
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
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
MCol *mcol = data;
|
|
|
|
MCol col[4];
|
|
|
|
int j;
|
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
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (j = 0; j < 4; ++j) {
|
2019-04-17 06:17:24 +02:00
|
|
|
col[j] = mcol[corner_indices[j]];
|
2019-04-22 09:39:35 +10: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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(mcol, col, sizeof(col));
|
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 void layerDefault_mcol(void *data, int count)
|
2006-11-11 16:38:37 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
static MCol default_mcol = {255, 255, 255, 255};
|
|
|
|
MCol *mcol = (MCol *)data;
|
|
|
|
int i;
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < 4 * count; i++) {
|
|
|
|
mcol[i] = default_mcol;
|
|
|
|
}
|
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
|
|
|
}
|
2006-11-11 16:38:37 +00:00
|
|
|
|
2013-09-24 03:58:19 +00:00
|
|
|
static void layerDefault_origindex(void *data, int count)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
copy_vn_i((int *)data, count, ORIGINDEX_NONE);
|
2013-09-24 03:58:19 +00:00
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
static void layerInterp_bweight(const void **sources,
|
|
|
|
const float *weights,
|
|
|
|
const float *UNUSED(sub_weights),
|
|
|
|
int count,
|
|
|
|
void *dest)
|
2010-07-14 22:06:10 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
float f;
|
|
|
|
float **in = (float **)sources;
|
|
|
|
int i;
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (count <= 0) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2010-07-14 22:06:10 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
f = 0.0f;
|
2011-12-28 13:11:46 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (weights) {
|
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
f += *in[i] * weights[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
f += *in[i];
|
|
|
|
}
|
|
|
|
}
|
2012-08-24 17:01:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* delay writing to the destination incase dest is in sources */
|
|
|
|
*((float *)dest) = f;
|
2010-07-14 22:06:10 +00:00
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
static void layerInterp_shapekey(const void **sources,
|
|
|
|
const float *weights,
|
|
|
|
const float *UNUSED(sub_weights),
|
|
|
|
int count,
|
|
|
|
void *dest)
|
2009-11-01 00:06:53 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
float co[3];
|
|
|
|
float **in = (float **)sources;
|
|
|
|
int i;
|
2008-07-08 02:22:37 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (count <= 0) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2009-11-01 00:06:53 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
zero_v3(co);
|
2011-12-28 13:11:46 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (weights) {
|
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
madd_v3_v3fl(co, in[i], weights[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
for (i = 0; i < count; ++i) {
|
|
|
|
add_v3_v3(co, in[i]);
|
|
|
|
}
|
|
|
|
}
|
2012-08-24 17:01:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* delay writing to the destination incase dest is in sources */
|
|
|
|
copy_v3_v3((float *)dest, co);
|
2009-11-01 00:06:53 +00:00
|
|
|
}
|
2008-07-08 02:22:37 +00:00
|
|
|
|
2012-05-22 15:18:43 +00:00
|
|
|
static void layerDefault_mvert_skin(void *data, int count)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
MVertSkin *vs = data;
|
|
|
|
int i;
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
copy_v3_fl(vs[i].radius, 0.25f);
|
|
|
|
vs[i].flag = 0;
|
|
|
|
}
|
2012-05-22 15:18:43 +00:00
|
|
|
}
|
2012-05-22 22:03:41 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
static void layerCopy_mvert_skin(const void *source, void *dest, int count)
|
2018-12-12 15:06:39 +01:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(dest, source, sizeof(MVertSkin) * count);
|
2018-12-12 15:06:39 +01:00
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
static void layerInterp_mvert_skin(const void **sources,
|
|
|
|
const float *weights,
|
|
|
|
const float *UNUSED(sub_weights),
|
|
|
|
int count,
|
|
|
|
void *dest)
|
2012-05-22 15:18:43 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
MVertSkin *vs_dst = dest;
|
|
|
|
float radius[3], w;
|
|
|
|
int i;
|
2012-05-22 15:18:43 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
zero_v3(radius);
|
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
const MVertSkin *vs_src = sources[i];
|
|
|
|
w = weights ? weights[i] : 1.0f;
|
2012-05-22 15:18:43 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
madd_v3_v3fl(radius, vs_src->radius, w);
|
|
|
|
}
|
2012-05-22 15:18:43 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* delay writing to the destination incase dest is in sources */
|
|
|
|
vs_dst = dest;
|
|
|
|
copy_v3_v3(vs_dst->radius, radius);
|
|
|
|
vs_dst->flag &= ~MVERT_SKIN_ROOT;
|
2012-05-22 15:18:43 +00:00
|
|
|
}
|
|
|
|
|
2014-04-16 16:51:49 +02:00
|
|
|
static void layerSwap_flnor(void *data, const int *corner_indices)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
short(*flnors)[4][3] = data;
|
|
|
|
short nors[4][3];
|
|
|
|
int i = 4;
|
2014-04-16 16:51:49 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
while (i--) {
|
|
|
|
copy_v3_v3_short(nors[i], (*flnors)[corner_indices[i]]);
|
|
|
|
}
|
2014-04-16 16:51:49 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(flnors, nors, sizeof(nors));
|
2014-04-16 16:51:49 +02:00
|
|
|
}
|
|
|
|
|
2017-05-31 00:18:20 +10:00
|
|
|
static void layerDefault_fmap(void *data, int count)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int *fmap_num = (int *)data;
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
fmap_num[i] = -1;
|
|
|
|
}
|
2017-05-31 00:18:20 +10:00
|
|
|
}
|
|
|
|
|
2011-02-13 10:52:18 +00:00
|
|
|
static const LayerTypeInfo LAYERTYPEINFO[CD_NUMTYPES] = {
|
2019-04-17 06:17:24 +02:00
|
|
|
/* 0: CD_MVERT */
|
|
|
|
{sizeof(MVert), "MVert", 1, NULL, NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 1: CD_MSTICKY */ /* DEPRECATED */
|
|
|
|
{sizeof(float) * 2, "", 1, NULL, NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 2: CD_MDEFORMVERT */
|
|
|
|
{sizeof(MDeformVert),
|
|
|
|
"MDeformVert",
|
|
|
|
1,
|
|
|
|
NULL,
|
|
|
|
layerCopy_mdeformvert,
|
|
|
|
layerFree_mdeformvert,
|
|
|
|
layerInterp_mdeformvert,
|
|
|
|
NULL,
|
|
|
|
NULL},
|
|
|
|
/* 3: CD_MEDGE */
|
|
|
|
{sizeof(MEdge), "MEdge", 1, NULL, NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 4: CD_MFACE */
|
|
|
|
{sizeof(MFace), "MFace", 1, NULL, NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 5: CD_MTFACE */
|
|
|
|
{sizeof(MTFace),
|
|
|
|
"MTFace",
|
|
|
|
1,
|
|
|
|
N_("UVMap"),
|
|
|
|
layerCopy_tface,
|
|
|
|
NULL,
|
|
|
|
layerInterp_tface,
|
|
|
|
layerSwap_tface,
|
|
|
|
layerDefault_tface,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerMaxNum_tface},
|
|
|
|
/* 6: CD_MCOL */
|
|
|
|
/* 4 MCol structs per face */
|
|
|
|
{sizeof(MCol) * 4,
|
|
|
|
"MCol",
|
|
|
|
4,
|
|
|
|
N_("Col"),
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerInterp_mcol,
|
|
|
|
layerSwap_mcol,
|
|
|
|
layerDefault_mcol,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerMaxNum_mloopcol},
|
|
|
|
/* 7: CD_ORIGINDEX */
|
|
|
|
{sizeof(int), "", 0, NULL, NULL, NULL, NULL, NULL, layerDefault_origindex},
|
|
|
|
/* 8: CD_NORMAL */
|
|
|
|
/* 3 floats per normal vector */
|
|
|
|
{sizeof(float) * 3,
|
|
|
|
"vec3f",
|
|
|
|
1,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerInterp_normal,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerValidate_normal,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerCopyValue_normal},
|
|
|
|
/* 9: CD_FACEMAP */
|
|
|
|
{sizeof(int), "", 0, NULL, NULL, NULL, NULL, NULL, layerDefault_fmap, NULL},
|
|
|
|
/* 10: CD_PROP_FLT */
|
|
|
|
{sizeof(MFloatProperty),
|
|
|
|
"MFloatProperty",
|
|
|
|
1,
|
|
|
|
N_("Float"),
|
|
|
|
layerCopy_propFloat,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerValidate_propFloat},
|
|
|
|
/* 11: CD_PROP_INT */
|
|
|
|
{sizeof(MIntProperty), "MIntProperty", 1, N_("Int"), layerCopy_propInt, NULL, NULL, NULL},
|
|
|
|
/* 12: CD_PROP_STR */
|
|
|
|
{sizeof(MStringProperty),
|
|
|
|
"MStringProperty",
|
|
|
|
1,
|
|
|
|
N_("String"),
|
|
|
|
layerCopy_propString,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL},
|
|
|
|
/* 13: CD_ORIGSPACE */
|
|
|
|
{sizeof(OrigSpaceFace),
|
|
|
|
"OrigSpaceFace",
|
|
|
|
1,
|
|
|
|
N_("UVMap"),
|
|
|
|
layerCopy_origspace_face,
|
|
|
|
NULL,
|
|
|
|
layerInterp_origspace_face,
|
|
|
|
layerSwap_origspace_face,
|
|
|
|
layerDefault_origspace_face},
|
|
|
|
/* 14: CD_ORCO */
|
|
|
|
{sizeof(float) * 3, "", 0, NULL, NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 15: CD_MTEXPOLY */ /* DEPRECATED */
|
2019-04-27 12:07:07 +10:00
|
|
|
/* note, when we expose the UV Map / TexFace split to the user,
|
|
|
|
* change this back to face Texture. */
|
2019-04-17 06:17:24 +02:00
|
|
|
{sizeof(int), "", 0, NULL, NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 16: CD_MLOOPUV */
|
|
|
|
{sizeof(MLoopUV),
|
|
|
|
"MLoopUV",
|
|
|
|
1,
|
|
|
|
N_("UVMap"),
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerInterp_mloopuv,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerValidate_mloopuv,
|
|
|
|
layerEqual_mloopuv,
|
|
|
|
layerMultiply_mloopuv,
|
|
|
|
layerInitMinMax_mloopuv,
|
|
|
|
layerAdd_mloopuv,
|
|
|
|
layerDoMinMax_mloopuv,
|
|
|
|
layerCopyValue_mloopuv,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerMaxNum_tface},
|
|
|
|
/* 17: CD_MLOOPCOL */
|
|
|
|
{sizeof(MLoopCol),
|
|
|
|
"MLoopCol",
|
|
|
|
1,
|
|
|
|
N_("Col"),
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerInterp_mloopcol,
|
|
|
|
NULL,
|
|
|
|
layerDefault_mloopcol,
|
|
|
|
NULL,
|
|
|
|
layerEqual_mloopcol,
|
|
|
|
layerMultiply_mloopcol,
|
|
|
|
layerInitMinMax_mloopcol,
|
|
|
|
layerAdd_mloopcol,
|
|
|
|
layerDoMinMax_mloopcol,
|
|
|
|
layerCopyValue_mloopcol,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerMaxNum_mloopcol},
|
|
|
|
/* 18: CD_TANGENT */
|
|
|
|
{sizeof(float) * 4 * 4, "", 0, N_("Tangent"), NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 19: CD_MDISPS */
|
|
|
|
{sizeof(MDisps),
|
|
|
|
"MDisps",
|
|
|
|
1,
|
|
|
|
NULL,
|
|
|
|
layerCopy_mdisps,
|
|
|
|
layerFree_mdisps,
|
|
|
|
NULL,
|
|
|
|
layerSwap_mdisps,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerRead_mdisps,
|
|
|
|
layerWrite_mdisps,
|
|
|
|
layerFilesize_mdisps},
|
|
|
|
/* 20: CD_PREVIEW_MCOL */
|
|
|
|
{sizeof(MCol) * 4,
|
|
|
|
"MCol",
|
|
|
|
4,
|
|
|
|
N_("PreviewCol"),
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerInterp_mcol,
|
|
|
|
layerSwap_mcol,
|
|
|
|
layerDefault_mcol},
|
|
|
|
/* 21: CD_ID_MCOL */ /* DEPRECATED */
|
|
|
|
{sizeof(MCol) * 4, "", 0, NULL, NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 22: CD_TEXTURE_MCOL */
|
|
|
|
{sizeof(MCol) * 4,
|
|
|
|
"MCol",
|
|
|
|
4,
|
|
|
|
N_("TexturedCol"),
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerInterp_mcol,
|
|
|
|
layerSwap_mcol,
|
|
|
|
layerDefault_mcol},
|
|
|
|
/* 23: CD_CLOTH_ORCO */
|
|
|
|
{sizeof(float) * 3, "", 0, NULL, NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 24: CD_RECAST */
|
|
|
|
{sizeof(MRecast), "MRecast", 1, N_("Recast"), NULL, NULL, NULL, NULL},
|
|
|
|
|
|
|
|
/* BMESH ONLY */
|
|
|
|
/* 25: CD_MPOLY */
|
|
|
|
{sizeof(MPoly), "MPoly", 1, N_("NGon Face"), NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 26: CD_MLOOP */
|
|
|
|
{sizeof(MLoop), "MLoop", 1, N_("NGon Face-Vertex"), NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 27: CD_SHAPE_KEYINDEX */
|
|
|
|
{sizeof(int), "", 0, NULL, NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 28: CD_SHAPEKEY */
|
|
|
|
{sizeof(float) * 3, "", 0, N_("ShapeKey"), NULL, NULL, layerInterp_shapekey},
|
|
|
|
/* 29: CD_BWEIGHT */
|
|
|
|
{sizeof(float), "", 0, N_("BevelWeight"), NULL, NULL, layerInterp_bweight},
|
|
|
|
/* 30: CD_CREASE */
|
|
|
|
{sizeof(float), "", 0, N_("SubSurfCrease"), NULL, NULL, layerInterp_bweight},
|
|
|
|
/* 31: CD_ORIGSPACE_MLOOP */
|
|
|
|
{sizeof(OrigSpaceLoop),
|
|
|
|
"OrigSpaceLoop",
|
|
|
|
1,
|
|
|
|
N_("OS Loop"),
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerInterp_mloop_origspace,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerEqual_mloop_origspace,
|
|
|
|
layerMultiply_mloop_origspace,
|
|
|
|
layerInitMinMax_mloop_origspace,
|
|
|
|
layerAdd_mloop_origspace,
|
|
|
|
layerDoMinMax_mloop_origspace,
|
|
|
|
layerCopyValue_mloop_origspace},
|
|
|
|
/* 32: CD_PREVIEW_MLOOPCOL */
|
|
|
|
{sizeof(MLoopCol),
|
|
|
|
"MLoopCol",
|
|
|
|
1,
|
|
|
|
N_("PreviewLoopCol"),
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
layerInterp_mloopcol,
|
|
|
|
NULL,
|
|
|
|
layerDefault_mloopcol,
|
|
|
|
NULL,
|
|
|
|
layerEqual_mloopcol,
|
|
|
|
layerMultiply_mloopcol,
|
|
|
|
layerInitMinMax_mloopcol,
|
|
|
|
layerAdd_mloopcol,
|
|
|
|
layerDoMinMax_mloopcol,
|
|
|
|
layerCopyValue_mloopcol},
|
|
|
|
/* 33: CD_BM_ELEM_PYPTR */
|
|
|
|
{sizeof(void *),
|
|
|
|
"",
|
|
|
|
1,
|
|
|
|
NULL,
|
|
|
|
layerCopy_bmesh_elem_py_ptr,
|
|
|
|
layerFree_bmesh_elem_py_ptr,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL},
|
|
|
|
|
|
|
|
/* END BMESH ONLY */
|
|
|
|
|
|
|
|
/* 34: CD_PAINT_MASK */
|
|
|
|
{sizeof(float), "", 0, NULL, NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 35: CD_GRID_PAINT_MASK */
|
|
|
|
{sizeof(GridPaintMask),
|
|
|
|
"GridPaintMask",
|
|
|
|
1,
|
|
|
|
NULL,
|
|
|
|
layerCopy_grid_paint_mask,
|
|
|
|
layerFree_grid_paint_mask,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL},
|
|
|
|
/* 36: CD_MVERT_SKIN */
|
|
|
|
{sizeof(MVertSkin),
|
|
|
|
"MVertSkin",
|
|
|
|
1,
|
|
|
|
NULL,
|
|
|
|
layerCopy_mvert_skin,
|
|
|
|
NULL,
|
|
|
|
layerInterp_mvert_skin,
|
|
|
|
NULL,
|
|
|
|
layerDefault_mvert_skin},
|
|
|
|
/* 37: CD_FREESTYLE_EDGE */
|
|
|
|
{sizeof(FreestyleEdge), "FreestyleEdge", 1, NULL, NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 38: CD_FREESTYLE_FACE */
|
|
|
|
{sizeof(FreestyleFace), "FreestyleFace", 1, NULL, NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 39: CD_MLOOPTANGENT */
|
|
|
|
{sizeof(float[4]), "", 0, NULL, NULL, NULL, NULL, NULL, NULL},
|
|
|
|
/* 40: CD_TESSLOOPNORMAL */
|
|
|
|
{sizeof(short[4][3]), "", 0, NULL, NULL, NULL, NULL, layerSwap_flnor, NULL},
|
|
|
|
/* 41: CD_CUSTOMLOOPNORMAL */
|
|
|
|
{sizeof(short[2]), "vec2s", 1, NULL, NULL, NULL, NULL, NULL, NULL},
|
2006-11-11 16:38:37 +00:00
|
|
|
};
|
|
|
|
|
2011-02-13 10:52:18 +00:00
|
|
|
static const char *LAYERTYPENAMES[CD_NUMTYPES] = {
|
2019-04-17 06:17:24 +02:00
|
|
|
/* 0-4 */ "CDMVert",
|
|
|
|
"CDMSticky",
|
|
|
|
"CDMDeformVert",
|
|
|
|
"CDMEdge",
|
|
|
|
"CDMFace",
|
|
|
|
/* 5-9 */ "CDMTFace",
|
|
|
|
"CDMCol",
|
|
|
|
"CDOrigIndex",
|
|
|
|
"CDNormal",
|
|
|
|
"CDFaceMap",
|
|
|
|
/* 10-14 */ "CDMFloatProperty",
|
|
|
|
"CDMIntProperty",
|
|
|
|
"CDMStringProperty",
|
|
|
|
"CDOrigSpace",
|
|
|
|
"CDOrco",
|
|
|
|
/* 15-19 */ "CDMTexPoly",
|
|
|
|
"CDMLoopUV",
|
|
|
|
"CDMloopCol",
|
|
|
|
"CDTangent",
|
|
|
|
"CDMDisps",
|
|
|
|
/* 20-24 */ "CDPreviewMCol",
|
|
|
|
"CDIDMCol",
|
|
|
|
"CDTextureMCol",
|
|
|
|
"CDClothOrco",
|
|
|
|
"CDMRecast",
|
|
|
|
|
|
|
|
/* BMESH ONLY */
|
|
|
|
/* 25-29 */ "CDMPoly",
|
|
|
|
"CDMLoop",
|
|
|
|
"CDShapeKeyIndex",
|
|
|
|
"CDShapeKey",
|
|
|
|
"CDBevelWeight",
|
|
|
|
/* 30-34 */ "CDSubSurfCrease",
|
|
|
|
"CDOrigSpaceLoop",
|
|
|
|
"CDPreviewLoopCol",
|
|
|
|
"CDBMElemPyPtr",
|
|
|
|
"CDPaintMask",
|
|
|
|
/* 35-36 */ "CDGridPaintMask",
|
|
|
|
"CDMVertSkin",
|
|
|
|
/* 37-38 */ "CDFreestyleEdge",
|
|
|
|
"CDFreestyleFace",
|
|
|
|
/* 39-41 */ "CDMLoopTangent",
|
|
|
|
"CDTessLoopNormal",
|
|
|
|
"CDCustomLoopNormal",
|
2010-05-27 08:42:59 +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
|
|
|
|
Refactor CDData masks, to have one mask per mesh elem type.
We already have different storages for cddata of verts, edges etc.,
'simply' do the same for the mask flags we use all around Blender code
to request some data, or limit some operation to some layers, etc.
Reason we need this is that some cddata types (like Normals) are
actually shared between verts/polys/loops, and we don’t want to generate
clnors everytime we request vnors!
As a side note, this also does final fix to T59338, which was the
trigger for this patch (need to request computed loop normals for
another mesh than evaluated one).
Reviewers: brecht, campbellbarton, sergey
Differential Revision: https://developer.blender.org/D4407
2019-03-07 11:13:40 +01:00
|
|
|
const CustomData_MeshMasks CD_MASK_BAREMESH = {
|
|
|
|
.vmask = CD_MASK_MVERT | CD_MASK_BWEIGHT,
|
|
|
|
.emask = CD_MASK_MEDGE | CD_MASK_BWEIGHT,
|
|
|
|
.fmask = 0,
|
|
|
|
.lmask = CD_MASK_MLOOP,
|
|
|
|
.pmask = CD_MASK_MPOLY | CD_MASK_FACEMAP,
|
|
|
|
};
|
|
|
|
const CustomData_MeshMasks CD_MASK_BAREMESH_ORIGINDEX = {
|
|
|
|
.vmask = CD_MASK_MVERT | CD_MASK_BWEIGHT | CD_MASK_ORIGINDEX,
|
|
|
|
.emask = CD_MASK_MEDGE | CD_MASK_BWEIGHT | CD_MASK_ORIGINDEX,
|
|
|
|
.fmask = 0,
|
|
|
|
.lmask = CD_MASK_MLOOP,
|
|
|
|
.pmask = CD_MASK_MPOLY | CD_MASK_FACEMAP | CD_MASK_ORIGINDEX,
|
|
|
|
};
|
|
|
|
const CustomData_MeshMasks CD_MASK_MESH = {
|
2019-04-17 06:17:24 +02:00
|
|
|
.vmask = (CD_MASK_MVERT | CD_MASK_MDEFORMVERT | CD_MASK_MVERT_SKIN | CD_MASK_PAINT_MASK |
|
|
|
|
CD_MASK_GENERIC_DATA),
|
2019-03-08 15:54:39 +01:00
|
|
|
.emask = (CD_MASK_MEDGE | CD_MASK_FREESTYLE_EDGE | CD_MASK_GENERIC_DATA),
|
Refactor CDData masks, to have one mask per mesh elem type.
We already have different storages for cddata of verts, edges etc.,
'simply' do the same for the mask flags we use all around Blender code
to request some data, or limit some operation to some layers, etc.
Reason we need this is that some cddata types (like Normals) are
actually shared between verts/polys/loops, and we don’t want to generate
clnors everytime we request vnors!
As a side note, this also does final fix to T59338, which was the
trigger for this patch (need to request computed loop normals for
another mesh than evaluated one).
Reviewers: brecht, campbellbarton, sergey
Differential Revision: https://developer.blender.org/D4407
2019-03-07 11:13:40 +01:00
|
|
|
.fmask = 0,
|
2019-04-17 06:17:24 +02:00
|
|
|
.lmask = (CD_MASK_MLOOP | CD_MASK_MDISPS | CD_MASK_MLOOPUV | CD_MASK_MLOOPCOL |
|
|
|
|
CD_MASK_CUSTOMLOOPNORMAL | CD_MASK_GRID_PAINT_MASK | CD_MASK_GENERIC_DATA),
|
|
|
|
.pmask = (CD_MASK_MPOLY | CD_MASK_RECAST | CD_MASK_FACEMAP | CD_MASK_FREESTYLE_FACE |
|
|
|
|
CD_MASK_GENERIC_DATA),
|
Refactor CDData masks, to have one mask per mesh elem type.
We already have different storages for cddata of verts, edges etc.,
'simply' do the same for the mask flags we use all around Blender code
to request some data, or limit some operation to some layers, etc.
Reason we need this is that some cddata types (like Normals) are
actually shared between verts/polys/loops, and we don’t want to generate
clnors everytime we request vnors!
As a side note, this also does final fix to T59338, which was the
trigger for this patch (need to request computed loop normals for
another mesh than evaluated one).
Reviewers: brecht, campbellbarton, sergey
Differential Revision: https://developer.blender.org/D4407
2019-03-07 11:13:40 +01:00
|
|
|
};
|
|
|
|
const CustomData_MeshMasks CD_MASK_EDITMESH = {
|
|
|
|
.vmask = (CD_MASK_MDEFORMVERT | CD_MASK_PAINT_MASK | CD_MASK_MVERT_SKIN | CD_MASK_SHAPEKEY |
|
|
|
|
CD_MASK_SHAPE_KEYINDEX | CD_MASK_GENERIC_DATA),
|
|
|
|
.emask = (CD_MASK_GENERIC_DATA),
|
|
|
|
.fmask = 0,
|
2019-03-08 15:54:39 +01:00
|
|
|
.lmask = (CD_MASK_MDISPS | CD_MASK_MLOOPUV | CD_MASK_MLOOPCOL | CD_MASK_CUSTOMLOOPNORMAL |
|
|
|
|
CD_MASK_GRID_PAINT_MASK | CD_MASK_GENERIC_DATA),
|
|
|
|
.pmask = (CD_MASK_RECAST | CD_MASK_FACEMAP | CD_MASK_GENERIC_DATA),
|
Refactor CDData masks, to have one mask per mesh elem type.
We already have different storages for cddata of verts, edges etc.,
'simply' do the same for the mask flags we use all around Blender code
to request some data, or limit some operation to some layers, etc.
Reason we need this is that some cddata types (like Normals) are
actually shared between verts/polys/loops, and we don’t want to generate
clnors everytime we request vnors!
As a side note, this also does final fix to T59338, which was the
trigger for this patch (need to request computed loop normals for
another mesh than evaluated one).
Reviewers: brecht, campbellbarton, sergey
Differential Revision: https://developer.blender.org/D4407
2019-03-07 11:13:40 +01:00
|
|
|
};
|
|
|
|
const CustomData_MeshMasks CD_MASK_DERIVEDMESH = {
|
|
|
|
.vmask = (CD_MASK_ORIGINDEX | CD_MASK_MDEFORMVERT | CD_MASK_SHAPEKEY | CD_MASK_MVERT_SKIN |
|
|
|
|
CD_MASK_ORCO | CD_MASK_CLOTH_ORCO | CD_MASK_GENERIC_DATA),
|
|
|
|
.emask = (CD_MASK_ORIGINDEX | CD_MASK_FREESTYLE_EDGE | CD_MASK_GENERIC_DATA),
|
|
|
|
.fmask = (CD_MASK_ORIGINDEX | CD_MASK_ORIGSPACE | CD_MASK_PREVIEW_MCOL | CD_MASK_TANGENT),
|
2019-04-17 06:17:24 +02:00
|
|
|
.lmask = (CD_MASK_MLOOPUV | CD_MASK_MLOOPCOL | CD_MASK_CUSTOMLOOPNORMAL |
|
|
|
|
CD_MASK_PREVIEW_MLOOPCOL | CD_MASK_ORIGSPACE_MLOOP |
|
|
|
|
CD_MASK_GENERIC_DATA), /* XXX MISSING CD_MASK_MLOOPTANGENT ? */
|
|
|
|
.pmask = (CD_MASK_ORIGINDEX | CD_MASK_RECAST | CD_MASK_FREESTYLE_FACE | CD_MASK_FACEMAP |
|
|
|
|
CD_MASK_GENERIC_DATA),
|
Refactor CDData masks, to have one mask per mesh elem type.
We already have different storages for cddata of verts, edges etc.,
'simply' do the same for the mask flags we use all around Blender code
to request some data, or limit some operation to some layers, etc.
Reason we need this is that some cddata types (like Normals) are
actually shared between verts/polys/loops, and we don’t want to generate
clnors everytime we request vnors!
As a side note, this also does final fix to T59338, which was the
trigger for this patch (need to request computed loop normals for
another mesh than evaluated one).
Reviewers: brecht, campbellbarton, sergey
Differential Revision: https://developer.blender.org/D4407
2019-03-07 11:13:40 +01:00
|
|
|
};
|
|
|
|
const CustomData_MeshMasks CD_MASK_BMESH = {
|
|
|
|
.vmask = (CD_MASK_MDEFORMVERT | CD_MASK_BWEIGHT | CD_MASK_MVERT_SKIN | CD_MASK_SHAPEKEY |
|
|
|
|
CD_MASK_SHAPE_KEYINDEX | CD_MASK_PAINT_MASK | CD_MASK_GENERIC_DATA),
|
|
|
|
.emask = (CD_MASK_BWEIGHT | CD_MASK_CREASE | CD_MASK_FREESTYLE_EDGE | CD_MASK_GENERIC_DATA),
|
|
|
|
.fmask = 0,
|
|
|
|
.lmask = (CD_MASK_MDISPS | CD_MASK_MLOOPUV | CD_MASK_MLOOPCOL | CD_MASK_CUSTOMLOOPNORMAL |
|
|
|
|
CD_MASK_GRID_PAINT_MASK | CD_MASK_GENERIC_DATA),
|
|
|
|
.pmask = (CD_MASK_RECAST | CD_MASK_FREESTYLE_FACE | CD_MASK_FACEMAP | CD_MASK_GENERIC_DATA),
|
|
|
|
};
|
2015-09-23 22:57:00 +10:00
|
|
|
/**
|
|
|
|
* cover values copied by #BKE_mesh_loops_to_tessdata
|
|
|
|
*/
|
Refactor CDData masks, to have one mask per mesh elem type.
We already have different storages for cddata of verts, edges etc.,
'simply' do the same for the mask flags we use all around Blender code
to request some data, or limit some operation to some layers, etc.
Reason we need this is that some cddata types (like Normals) are
actually shared between verts/polys/loops, and we don’t want to generate
clnors everytime we request vnors!
As a side note, this also does final fix to T59338, which was the
trigger for this patch (need to request computed loop normals for
another mesh than evaluated one).
Reviewers: brecht, campbellbarton, sergey
Differential Revision: https://developer.blender.org/D4407
2019-03-07 11:13:40 +01:00
|
|
|
const CustomData_MeshMasks CD_MASK_FACECORNERS = {
|
|
|
|
.vmask = 0,
|
|
|
|
.emask = 0,
|
|
|
|
.fmask = (CD_MASK_MTFACE | CD_MASK_MCOL | CD_MASK_PREVIEW_MCOL | CD_MASK_ORIGSPACE |
|
|
|
|
CD_MASK_TESSLOOPNORMAL | CD_MASK_TANGENT),
|
2019-04-17 06:17:24 +02:00
|
|
|
.lmask = (CD_MASK_MLOOPUV | CD_MASK_MLOOPCOL | CD_MASK_PREVIEW_MLOOPCOL |
|
|
|
|
CD_MASK_ORIGSPACE_MLOOP | CD_MASK_NORMAL | CD_MASK_MLOOPTANGENT),
|
Refactor CDData masks, to have one mask per mesh elem type.
We already have different storages for cddata of verts, edges etc.,
'simply' do the same for the mask flags we use all around Blender code
to request some data, or limit some operation to some layers, etc.
Reason we need this is that some cddata types (like Normals) are
actually shared between verts/polys/loops, and we don’t want to generate
clnors everytime we request vnors!
As a side note, this also does final fix to T59338, which was the
trigger for this patch (need to request computed loop normals for
another mesh than evaluated one).
Reviewers: brecht, campbellbarton, sergey
Differential Revision: https://developer.blender.org/D4407
2019-03-07 11:13:40 +01:00
|
|
|
.pmask = 0,
|
|
|
|
};
|
|
|
|
const CustomData_MeshMasks CD_MASK_EVERYTHING = {
|
2019-04-17 06:17:24 +02:00
|
|
|
.vmask = (CD_MASK_MVERT | CD_MASK_BM_ELEM_PYPTR | CD_MASK_ORIGINDEX | CD_MASK_NORMAL |
|
|
|
|
CD_MASK_MDEFORMVERT | CD_MASK_BWEIGHT | CD_MASK_MVERT_SKIN | CD_MASK_ORCO |
|
|
|
|
CD_MASK_CLOTH_ORCO | CD_MASK_SHAPEKEY | CD_MASK_SHAPE_KEYINDEX | CD_MASK_PAINT_MASK |
|
|
|
|
CD_MASK_GENERIC_DATA),
|
|
|
|
.emask = (CD_MASK_MEDGE | CD_MASK_BM_ELEM_PYPTR | CD_MASK_ORIGINDEX | CD_MASK_BWEIGHT |
|
|
|
|
CD_MASK_CREASE | CD_MASK_FREESTYLE_EDGE | CD_MASK_GENERIC_DATA),
|
Refactor CDData masks, to have one mask per mesh elem type.
We already have different storages for cddata of verts, edges etc.,
'simply' do the same for the mask flags we use all around Blender code
to request some data, or limit some operation to some layers, etc.
Reason we need this is that some cddata types (like Normals) are
actually shared between verts/polys/loops, and we don’t want to generate
clnors everytime we request vnors!
As a side note, this also does final fix to T59338, which was the
trigger for this patch (need to request computed loop normals for
another mesh than evaluated one).
Reviewers: brecht, campbellbarton, sergey
Differential Revision: https://developer.blender.org/D4407
2019-03-07 11:13:40 +01:00
|
|
|
.fmask = (CD_MASK_MFACE | CD_MASK_ORIGINDEX | CD_MASK_NORMAL | CD_MASK_MTFACE | CD_MASK_MCOL |
|
|
|
|
CD_MASK_ORIGSPACE | CD_MASK_TANGENT | CD_MASK_TESSLOOPNORMAL | CD_MASK_PREVIEW_MCOL |
|
|
|
|
CD_MASK_GENERIC_DATA),
|
2019-04-17 06:17:24 +02:00
|
|
|
.lmask = (CD_MASK_MLOOP | CD_MASK_BM_ELEM_PYPTR | CD_MASK_MDISPS | CD_MASK_NORMAL |
|
|
|
|
CD_MASK_MLOOPUV | CD_MASK_MLOOPCOL | CD_MASK_CUSTOMLOOPNORMAL |
|
|
|
|
CD_MASK_MLOOPTANGENT | CD_MASK_PREVIEW_MLOOPCOL | CD_MASK_ORIGSPACE_MLOOP |
|
|
|
|
CD_MASK_GRID_PAINT_MASK | CD_MASK_GENERIC_DATA),
|
|
|
|
.pmask = (CD_MASK_MPOLY | CD_MASK_BM_ELEM_PYPTR | CD_MASK_ORIGINDEX | CD_MASK_NORMAL |
|
|
|
|
CD_MASK_RECAST | CD_MASK_FACEMAP | CD_MASK_FREESTYLE_FACE | CD_MASK_GENERIC_DATA),
|
Refactor CDData masks, to have one mask per mesh elem type.
We already have different storages for cddata of verts, edges etc.,
'simply' do the same for the mask flags we use all around Blender code
to request some data, or limit some operation to some layers, etc.
Reason we need this is that some cddata types (like Normals) are
actually shared between verts/polys/loops, and we don’t want to generate
clnors everytime we request vnors!
As a side note, this also does final fix to T59338, which was the
trigger for this patch (need to request computed loop normals for
another mesh than evaluated one).
Reviewers: brecht, campbellbarton, sergey
Differential Revision: https://developer.blender.org/D4407
2019-03-07 11:13:40 +01:00
|
|
|
};
|
2008-07-08 02:22:37 +00:00
|
|
|
|
2006-11-11 16:38:37 +00:00
|
|
|
static const LayerTypeInfo *layerType_getInfo(int type)
|
|
|
|
{
|
2019-04-22 09:39:35 +10:00
|
|
|
if (type < 0 || type >= CD_NUMTYPES) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2006-11-11 16:38:37 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return &LAYERTYPEINFO[type];
|
2006-11-11 16:38:37 +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
|
|
|
static const char *layerType_getName(int type)
|
2006-08-28 01:12:36 +00:00
|
|
|
{
|
2019-04-22 09:39:35 +10:00
|
|
|
if (type < 0 || type >= CD_NUMTYPES) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
2019-04-22 09:39:35 +10: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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return LAYERTYPENAMES[type];
|
2006-08-28 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
Refactor CDData masks, to have one mask per mesh elem type.
We already have different storages for cddata of verts, edges etc.,
'simply' do the same for the mask flags we use all around Blender code
to request some data, or limit some operation to some layers, etc.
Reason we need this is that some cddata types (like Normals) are
actually shared between verts/polys/loops, and we don’t want to generate
clnors everytime we request vnors!
As a side note, this also does final fix to T59338, which was the
trigger for this patch (need to request computed loop normals for
another mesh than evaluated one).
Reviewers: brecht, campbellbarton, sergey
Differential Revision: https://developer.blender.org/D4407
2019-03-07 11:13:40 +01:00
|
|
|
void customData_mask_layers__print(const CustomData_MeshMasks *mask)
|
2012-05-28 21:02:44 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
|
|
|
|
|
|
|
printf("verts mask=0x%lx:\n", (long unsigned int)mask->vmask);
|
|
|
|
for (i = 0; i < CD_NUMTYPES; i++) {
|
|
|
|
if (mask->vmask & CD_TYPE_AS_MASK(i)) {
|
|
|
|
printf(" %s\n", layerType_getName(i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("edges mask=0x%lx:\n", (long unsigned int)mask->emask);
|
|
|
|
for (i = 0; i < CD_NUMTYPES; i++) {
|
|
|
|
if (mask->emask & CD_TYPE_AS_MASK(i)) {
|
|
|
|
printf(" %s\n", layerType_getName(i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("faces mask=0x%lx:\n", (long unsigned int)mask->fmask);
|
|
|
|
for (i = 0; i < CD_NUMTYPES; i++) {
|
|
|
|
if (mask->fmask & CD_TYPE_AS_MASK(i)) {
|
|
|
|
printf(" %s\n", layerType_getName(i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("loops mask=0x%lx:\n", (long unsigned int)mask->lmask);
|
|
|
|
for (i = 0; i < CD_NUMTYPES; i++) {
|
|
|
|
if (mask->lmask & CD_TYPE_AS_MASK(i)) {
|
|
|
|
printf(" %s\n", layerType_getName(i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("polys mask=0x%lx:\n", (long unsigned int)mask->pmask);
|
|
|
|
for (i = 0; i < CD_NUMTYPES; i++) {
|
|
|
|
if (mask->pmask & CD_TYPE_AS_MASK(i)) {
|
|
|
|
printf(" %s\n", layerType_getName(i));
|
|
|
|
}
|
|
|
|
}
|
2012-05-28 21:02:44 +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 functions *********************/
|
2006-12-12 21:29:09 +00:00
|
|
|
static void customData_update_offsets(CustomData *data);
|
2006-11-11 16:38:37 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
static CustomDataLayer *customData_add_layer__internal(CustomData *data,
|
|
|
|
int type,
|
|
|
|
eCDAllocType alloctype,
|
|
|
|
void *layerdata,
|
|
|
|
int totelem,
|
|
|
|
const char *name);
|
2006-11-11 16:38:37 +00:00
|
|
|
|
2011-05-11 02:14:43 +00:00
|
|
|
void CustomData_update_typemap(CustomData *data)
|
2009-09-15 15:32:09 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i, lasttype = -1;
|
2009-09-15 15:32:09 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < CD_NUMTYPES; i++) {
|
|
|
|
data->typemap[i] = -1;
|
|
|
|
}
|
2009-09-15 15:32:09 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < data->totlayer; i++) {
|
|
|
|
const int type = data->layers[i].type;
|
|
|
|
if (type != lasttype) {
|
|
|
|
data->typemap[type] = i;
|
|
|
|
lasttype = type;
|
|
|
|
}
|
|
|
|
}
|
2009-09-15 15:32:09 +00:00
|
|
|
}
|
|
|
|
|
2013-02-04 16:20:39 +00:00
|
|
|
/* currently only used in BLI_assert */
|
|
|
|
#ifndef NDEBUG
|
2014-02-05 22:36:15 +11:00
|
|
|
static bool customdata_typemap_is_valid(const CustomData *data)
|
2012-10-31 09:50:24 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomData data_copy = *data;
|
|
|
|
CustomData_update_typemap(&data_copy);
|
|
|
|
return (memcmp(data->typemap, data_copy.typemap, sizeof(data->typemap)) == 0);
|
2012-10-31 09:50:24 +00:00
|
|
|
}
|
2013-02-04 16:20:39 +00:00
|
|
|
#endif
|
2012-10-31 09:50:24 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
bool CustomData_merge(const struct CustomData *source,
|
|
|
|
struct CustomData *dest,
|
|
|
|
CustomDataMask mask,
|
|
|
|
eCDAllocType alloctype,
|
|
|
|
int totelem)
|
|
|
|
{
|
|
|
|
/*const LayerTypeInfo *typeInfo;*/
|
|
|
|
CustomDataLayer *layer, *newlayer;
|
|
|
|
void *data;
|
|
|
|
int i, type, lasttype = -1, lastactive = 0, lastrender = 0, lastclone = 0, lastmask = 0,
|
|
|
|
flag = 0;
|
|
|
|
int number = 0, maxnumber = -1;
|
|
|
|
bool changed = false;
|
|
|
|
|
|
|
|
for (i = 0; i < source->totlayer; ++i) {
|
|
|
|
layer = &source->layers[i];
|
|
|
|
/*typeInfo = layerType_getInfo(layer->type);*/ /*UNUSED*/
|
|
|
|
|
|
|
|
type = layer->type;
|
|
|
|
flag = layer->flag;
|
|
|
|
|
|
|
|
if (type != lasttype) {
|
|
|
|
number = 0;
|
|
|
|
maxnumber = CustomData_layertype_layers_max(type);
|
|
|
|
lastactive = layer->active;
|
|
|
|
lastrender = layer->active_rnd;
|
|
|
|
lastclone = layer->active_clone;
|
|
|
|
lastmask = layer->active_mask;
|
|
|
|
lasttype = type;
|
|
|
|
}
|
2019-04-22 09:39:35 +10:00
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
number++;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (flag & CD_FLAG_NOCOPY) {
|
2019-04-17 06:17:24 +02:00
|
|
|
continue;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
else if (!(mask & CD_TYPE_AS_MASK(type))) {
|
2019-04-17 06:17:24 +02:00
|
|
|
continue;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
else if ((maxnumber != -1) && (number >= maxnumber)) {
|
2019-04-17 06:17:24 +02:00
|
|
|
continue;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-06-21 14:12:50 +02:00
|
|
|
else if (CustomData_get_named_layer_index(dest, type, layer->name) != -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
continue;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
switch (alloctype) {
|
|
|
|
case CD_ASSIGN:
|
|
|
|
case CD_REFERENCE:
|
|
|
|
case CD_DUPLICATE:
|
|
|
|
data = layer->data;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
data = NULL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((alloctype == CD_ASSIGN) && (flag & CD_FLAG_NOFREE)) {
|
|
|
|
newlayer = customData_add_layer__internal(
|
|
|
|
dest, type, CD_REFERENCE, data, totelem, layer->name);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
newlayer = customData_add_layer__internal(dest, type, alloctype, data, totelem, layer->name);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (newlayer) {
|
|
|
|
newlayer->uid = layer->uid;
|
|
|
|
|
|
|
|
newlayer->active = lastactive;
|
|
|
|
newlayer->active_rnd = lastrender;
|
|
|
|
newlayer->active_clone = lastclone;
|
|
|
|
newlayer->active_mask = lastmask;
|
|
|
|
newlayer->flag |= flag & (CD_FLAG_EXTERNAL | CD_FLAG_IN_MEMORY);
|
|
|
|
changed = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CustomData_update_typemap(dest);
|
|
|
|
return changed;
|
2006-08-28 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
2015-08-25 10:29:40 +02:00
|
|
|
/* NOTE: Take care of referenced layers by yourself! */
|
|
|
|
void CustomData_realloc(CustomData *data, int totelem)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
CustomDataLayer *layer = &data->layers[i];
|
|
|
|
const LayerTypeInfo *typeInfo;
|
|
|
|
if (layer->flag & CD_FLAG_NOFREE) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
typeInfo = layerType_getInfo(layer->type);
|
|
|
|
layer->data = MEM_reallocN(layer->data, (size_t)totelem * typeInfo->size);
|
|
|
|
}
|
2015-08-25 10:29:40 +02:00
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
void CustomData_copy(const struct CustomData *source,
|
|
|
|
struct CustomData *dest,
|
|
|
|
CustomDataMask mask,
|
|
|
|
eCDAllocType alloctype,
|
|
|
|
int totelem)
|
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
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomData_reset(dest);
|
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
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (source->external) {
|
2019-04-17 06:17:24 +02:00
|
|
|
dest->external = MEM_dupallocN(source->external);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2010-03-30 12:01:17 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomData_merge(source, dest, mask, alloctype, totelem);
|
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
|
|
|
}
|
|
|
|
|
2006-12-12 21:29:09 +00:00
|
|
|
static void customData_free_layer__internal(CustomDataLayer *layer, int totelem)
|
2006-08-28 01:12:36 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo;
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (!(layer->flag & CD_FLAG_NOFREE) && layer->data) {
|
|
|
|
typeInfo = layerType_getInfo(layer->type);
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->free) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->free(layer->data, totelem, typeInfo->size);
|
2019-04-22 09:39:35 +10: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
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer->data) {
|
2019-04-17 06:17:24 +02:00
|
|
|
MEM_freeN(layer->data);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02: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
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2009-11-25 14:27:50 +00:00
|
|
|
static void CustomData_external_free(CustomData *data)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
if (data->external) {
|
|
|
|
MEM_freeN(data->external);
|
|
|
|
data->external = NULL;
|
|
|
|
}
|
2009-11-25 14:27:50 +00:00
|
|
|
}
|
|
|
|
|
2012-10-31 09:50:24 +00:00
|
|
|
void CustomData_reset(CustomData *data)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
memset(data, 0, sizeof(*data));
|
|
|
|
copy_vn_i(data->typemap, CD_NUMTYPES, -1);
|
2012-10-31 09:50:24 +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
|
|
|
void CustomData_free(CustomData *data, int totelem)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
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
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
2019-04-17 06:17:24 +02:00
|
|
|
customData_free_layer__internal(&data->layers[i], totelem);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (data->layers) {
|
2019-04-17 06:17:24 +02:00
|
|
|
MEM_freeN(data->layers);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomData_external_free(data);
|
|
|
|
CustomData_reset(data);
|
2006-08-28 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
2015-02-19 00:00:23 +05:00
|
|
|
void CustomData_free_typemask(struct CustomData *data, int totelem, CustomDataMask mask)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2015-02-19 00:00:23 +05:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
CustomDataLayer *layer = &data->layers[i];
|
|
|
|
if (!(mask & CD_TYPE_AS_MASK(layer->type))) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
customData_free_layer__internal(layer, totelem);
|
|
|
|
}
|
2015-02-19 00:00:23 +05:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (data->layers) {
|
2019-04-17 06:17:24 +02:00
|
|
|
MEM_freeN(data->layers);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2015-02-19 00:00:23 +05:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomData_external_free(data);
|
|
|
|
CustomData_reset(data);
|
2015-02-19 00:00:23 +05:00
|
|
|
}
|
|
|
|
|
2006-12-12 21:29:09 +00:00
|
|
|
static void customData_update_offsets(CustomData *data)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo;
|
|
|
|
int i, offset = 0;
|
2006-12-12 21:29:09 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
typeInfo = layerType_getInfo(data->layers[i].type);
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[i].offset = offset;
|
|
|
|
offset += typeInfo->size;
|
|
|
|
}
|
2006-12-12 21:29:09 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
data->totsize = offset;
|
|
|
|
CustomData_update_typemap(data);
|
2006-12-12 21:29:09 +00:00
|
|
|
}
|
|
|
|
|
2012-10-31 09:50:24 +00:00
|
|
|
/* to use when we're in the middle of modifying layers */
|
|
|
|
static int CustomData_get_layer_index__notypemap(const CustomData *data, int type)
|
2006-11-11 16:38:37 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2006-11-11 16:38:37 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (data->layers[i].type == type) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return i;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
}
|
2006-11-11 16:38:37 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return -1;
|
2006-12-12 21:29:09 +00:00
|
|
|
}
|
|
|
|
|
2012-10-31 09:50:24 +00:00
|
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
/* index values to access the layers (offset from the layer start) */
|
|
|
|
|
|
|
|
int CustomData_get_layer_index(const CustomData *data, int type)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
BLI_assert(customdata_typemap_is_valid(data));
|
|
|
|
return data->typemap[type];
|
2012-10-31 09:50:24 +00:00
|
|
|
}
|
|
|
|
|
2009-11-02 06:31:23 +00:00
|
|
|
int CustomData_get_layer_index_n(const struct CustomData *data, int type, int n)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i = CustomData_get_layer_index(data, type);
|
2009-11-02 06:31:23 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (i != -1) {
|
|
|
|
BLI_assert(i + n < data->totlayer);
|
|
|
|
i = (data->layers[i + n].type == type) ? (i + n) : (-1);
|
|
|
|
}
|
2009-11-02 06:31:23 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return i;
|
2009-11-02 06:31:23 +00:00
|
|
|
}
|
|
|
|
|
2010-11-17 09:45:45 +00:00
|
|
|
int CustomData_get_named_layer_index(const CustomData *data, int type, const char *name)
|
2006-12-23 23:33:03 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2006-12-23 23:33:03 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (data->layers[i].type == type) {
|
|
|
|
if (STREQ(data->layers[i].name, name)) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return i;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2006-12-23 23:33:03 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return -1;
|
2006-12-23 23:33:03 +00:00
|
|
|
}
|
|
|
|
|
2006-12-13 00:50:02 +00:00
|
|
|
int CustomData_get_active_layer_index(const CustomData *data, int type)
|
2006-12-12 21:29:09 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const int layer_index = data->typemap[type];
|
|
|
|
BLI_assert(customdata_typemap_is_valid(data));
|
|
|
|
return (layer_index != -1) ? layer_index + data->layers[layer_index].active : -1;
|
2006-11-11 16:38:37 +00:00
|
|
|
}
|
|
|
|
|
2007-05-02 00:01:23 +00:00
|
|
|
int CustomData_get_render_layer_index(const CustomData *data, int type)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const int layer_index = data->typemap[type];
|
|
|
|
BLI_assert(customdata_typemap_is_valid(data));
|
|
|
|
return (layer_index != -1) ? layer_index + data->layers[layer_index].active_rnd : -1;
|
2007-05-02 00:01:23 +00:00
|
|
|
}
|
|
|
|
|
2008-12-14 17:32:24 +00:00
|
|
|
int CustomData_get_clone_layer_index(const CustomData *data, int type)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const int layer_index = data->typemap[type];
|
|
|
|
BLI_assert(customdata_typemap_is_valid(data));
|
|
|
|
return (layer_index != -1) ? layer_index + data->layers[layer_index].active_clone : -1;
|
2008-12-14 17:32:24 +00:00
|
|
|
}
|
|
|
|
|
2009-12-22 10:48:13 +00:00
|
|
|
int CustomData_get_stencil_layer_index(const CustomData *data, int type)
|
2008-12-14 17:32:24 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const int layer_index = data->typemap[type];
|
|
|
|
BLI_assert(customdata_typemap_is_valid(data));
|
|
|
|
return (layer_index != -1) ? layer_index + data->layers[layer_index].active_mask : -1;
|
2012-10-31 09:50:24 +00:00
|
|
|
}
|
2008-12-14 17:32:24 +00:00
|
|
|
|
2012-10-31 09:50:24 +00:00
|
|
|
/* -------------------------------------------------------------------- */
|
|
|
|
/* index values per layer type */
|
2008-12-14 17:32:24 +00:00
|
|
|
|
2013-03-24 12:13:13 +00:00
|
|
|
int CustomData_get_named_layer(const struct CustomData *data, int type, const char *name)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const int named_index = CustomData_get_named_layer_index(data, type, name);
|
|
|
|
const int layer_index = data->typemap[type];
|
|
|
|
BLI_assert(customdata_typemap_is_valid(data));
|
|
|
|
return (named_index != -1) ? named_index - layer_index : -1;
|
2013-03-24 12:13:13 +00:00
|
|
|
}
|
|
|
|
|
2007-09-18 19:39:25 +00:00
|
|
|
int CustomData_get_active_layer(const CustomData *data, int type)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const int layer_index = data->typemap[type];
|
|
|
|
BLI_assert(customdata_typemap_is_valid(data));
|
|
|
|
return (layer_index != -1) ? data->layers[layer_index].active : -1;
|
2007-09-18 19:39:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int CustomData_get_render_layer(const CustomData *data, int type)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const int layer_index = data->typemap[type];
|
|
|
|
BLI_assert(customdata_typemap_is_valid(data));
|
|
|
|
return (layer_index != -1) ? data->layers[layer_index].active_rnd : -1;
|
2007-09-18 19:39:25 +00:00
|
|
|
}
|
|
|
|
|
2008-12-14 17:32:24 +00:00
|
|
|
int CustomData_get_clone_layer(const CustomData *data, int type)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const int layer_index = data->typemap[type];
|
|
|
|
BLI_assert(customdata_typemap_is_valid(data));
|
|
|
|
return (layer_index != -1) ? data->layers[layer_index].active_clone : -1;
|
2008-12-14 17:32:24 +00:00
|
|
|
}
|
|
|
|
|
2009-12-22 10:48:13 +00:00
|
|
|
int CustomData_get_stencil_layer(const CustomData *data, int type)
|
2008-12-14 17:32:24 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const int layer_index = data->typemap[type];
|
|
|
|
BLI_assert(customdata_typemap_is_valid(data));
|
|
|
|
return (layer_index != -1) ? data->layers[layer_index].active_mask : -1;
|
2008-12-14 17:32:24 +00:00
|
|
|
}
|
2007-09-18 19:39:25 +00:00
|
|
|
|
2006-12-12 21:29:09 +00:00
|
|
|
void CustomData_set_layer_active(CustomData *data, int type, int n)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2006-12-12 21:29:09 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (data->layers[i].type == type) {
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[i].active = n;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
}
|
2006-12-12 21:29:09 +00:00
|
|
|
}
|
|
|
|
|
2007-05-02 00:01:23 +00:00
|
|
|
void CustomData_set_layer_render(CustomData *data, int type, int n)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2007-05-02 00:01:23 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (data->layers[i].type == type) {
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[i].active_rnd = n;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
}
|
2007-05-02 00:01:23 +00:00
|
|
|
}
|
|
|
|
|
2008-12-14 17:32:24 +00:00
|
|
|
void CustomData_set_layer_clone(CustomData *data, int type, int n)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2008-12-14 17:32:24 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (data->layers[i].type == type) {
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[i].active_clone = n;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
}
|
2008-12-14 17:32:24 +00:00
|
|
|
}
|
|
|
|
|
2009-12-22 10:48:13 +00:00
|
|
|
void CustomData_set_layer_stencil(CustomData *data, int type, int n)
|
2008-12-14 17:32:24 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2008-12-14 17:32:24 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (data->layers[i].type == type) {
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[i].active_mask = n;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
}
|
2008-12-14 17:32:24 +00:00
|
|
|
}
|
|
|
|
|
2019-04-27 12:07:07 +10:00
|
|
|
/* For using with an index from CustomData_get_active_layer_index and
|
|
|
|
* CustomData_get_render_layer_index. */
|
2007-09-12 02:13:35 +00:00
|
|
|
void CustomData_set_layer_active_index(CustomData *data, int type, int n)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2007-09-12 02:13:35 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (data->layers[i].type == type) {
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[i].active = n - i;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
}
|
2007-09-12 02:13:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CustomData_set_layer_render_index(CustomData *data, int type, int n)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2007-09-12 02:13:35 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (data->layers[i].type == type) {
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[i].active_rnd = n - i;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
}
|
2007-09-12 02:13:35 +00:00
|
|
|
}
|
|
|
|
|
2008-12-14 17:32:24 +00:00
|
|
|
void CustomData_set_layer_clone_index(CustomData *data, int type, int n)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2008-12-14 17:32:24 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (data->layers[i].type == type) {
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[i].active_clone = n - i;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
}
|
2008-12-14 17:32:24 +00:00
|
|
|
}
|
|
|
|
|
2009-12-22 10:48:13 +00:00
|
|
|
void CustomData_set_layer_stencil_index(CustomData *data, int type, int n)
|
2008-12-14 17:32:24 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2008-12-14 17:32:24 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (data->layers[i].type == type) {
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[i].active_mask = n - i;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
}
|
2008-12-14 17:32:24 +00:00
|
|
|
}
|
2007-05-02 00:01:23 +00:00
|
|
|
|
2006-12-12 21:29:09 +00:00
|
|
|
void CustomData_set_layer_flag(struct CustomData *data, int type, int flag)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2006-12-12 21:29:09 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (data->layers[i].type == type) {
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[i].flag |= flag;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
}
|
2006-12-12 21:29:09 +00:00
|
|
|
}
|
|
|
|
|
2019-02-28 18:03:01 +01:00
|
|
|
void CustomData_clear_layer_flag(struct CustomData *data, int type, int flag)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const int nflag = ~flag;
|
2019-02-28 18:03:01 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (int i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (data->layers[i].type == type) {
|
|
|
|
data->layers[i].flag &= nflag;
|
|
|
|
}
|
|
|
|
}
|
2019-02-28 18:03:01 +01:00
|
|
|
}
|
|
|
|
|
2006-11-11 16:38:37 +00:00
|
|
|
static int customData_resize(CustomData *data, int amount)
|
2006-08-28 01:12:36 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomDataLayer *tmp = MEM_calloc_arrayN(
|
|
|
|
(data->maxlayer + amount), sizeof(*tmp), "CustomData->layers");
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!tmp) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return 0;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
data->maxlayer += amount;
|
|
|
|
if (data->layers) {
|
|
|
|
memcpy(tmp, data->layers, sizeof(*tmp) * data->totlayer);
|
|
|
|
MEM_freeN(data->layers);
|
|
|
|
}
|
|
|
|
data->layers = tmp;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static CustomDataLayer *customData_add_layer__internal(CustomData *data,
|
|
|
|
int type,
|
|
|
|
eCDAllocType alloctype,
|
|
|
|
void *layerdata,
|
|
|
|
int totelem,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
|
|
|
int flag = 0, index = data->totlayer;
|
|
|
|
void *newlayerdata = NULL;
|
|
|
|
|
2019-06-12 09:04:10 +10:00
|
|
|
/* Passing a layer-data to copy from with an alloctype that won't copy is
|
2019-04-17 06:17:24 +02:00
|
|
|
* most likely a bug */
|
|
|
|
BLI_assert(!layerdata || (alloctype == CD_ASSIGN) || (alloctype == CD_DUPLICATE) ||
|
|
|
|
(alloctype == CD_REFERENCE));
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!typeInfo->defaultname && CustomData_has_layer(data, type)) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return &data->layers[CustomData_get_layer_index(data, type)];
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
if ((alloctype == CD_ASSIGN) || (alloctype == CD_REFERENCE)) {
|
|
|
|
newlayerdata = layerdata;
|
|
|
|
}
|
|
|
|
else if (totelem > 0 && typeInfo->size > 0) {
|
|
|
|
if (alloctype == CD_DUPLICATE && layerdata) {
|
|
|
|
newlayerdata = MEM_malloc_arrayN((size_t)totelem, typeInfo->size, layerType_getName(type));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
newlayerdata = MEM_calloc_arrayN((size_t)totelem, typeInfo->size, layerType_getName(type));
|
|
|
|
}
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!newlayerdata) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (alloctype == CD_DUPLICATE && layerdata) {
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->copy) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->copy(layerdata, newlayerdata, totelem);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(newlayerdata, layerdata, (size_t)totelem * typeInfo->size);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
}
|
|
|
|
else if (alloctype == CD_DEFAULT) {
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->set_default) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->set_default(newlayerdata, totelem);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
}
|
2019-04-22 09:39:35 +10:00
|
|
|
else if (alloctype == CD_REFERENCE) {
|
2019-04-17 06:17:24 +02:00
|
|
|
flag |= CD_FLAG_NOFREE;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
if (index >= data->maxlayer) {
|
|
|
|
if (!customData_resize(data, CUSTOMDATA_GROW)) {
|
2019-04-22 09:39:35 +10:00
|
|
|
if (newlayerdata != layerdata) {
|
2019-04-17 06:17:24 +02:00
|
|
|
MEM_freeN(newlayerdata);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
data->totlayer++;
|
|
|
|
|
|
|
|
/* keep layers ordered by type */
|
2019-04-22 09:39:35 +10:00
|
|
|
for (; index > 0 && data->layers[index - 1].type > type; --index) {
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[index] = data->layers[index - 1];
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
data->layers[index].type = type;
|
|
|
|
data->layers[index].flag = flag;
|
|
|
|
data->layers[index].data = newlayerdata;
|
|
|
|
|
|
|
|
/* Set default name if none exists. Note we only call DATA_() once
|
|
|
|
* we know there is a default name, to avoid overhead of locale lookups
|
|
|
|
* in the depsgraph. */
|
|
|
|
if (!name && typeInfo->defaultname) {
|
|
|
|
name = DATA_(typeInfo->defaultname);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (name) {
|
|
|
|
BLI_strncpy(data->layers[index].name, name, sizeof(data->layers[index].name));
|
|
|
|
CustomData_set_layer_unique_name(data, index);
|
|
|
|
}
|
2019-04-22 09:39:35 +10:00
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[index].name[0] = '\0';
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
if (index > 0 && data->layers[index - 1].type == type) {
|
|
|
|
data->layers[index].active = data->layers[index - 1].active;
|
|
|
|
data->layers[index].active_rnd = data->layers[index - 1].active_rnd;
|
|
|
|
data->layers[index].active_clone = data->layers[index - 1].active_clone;
|
|
|
|
data->layers[index].active_mask = data->layers[index - 1].active_mask;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
data->layers[index].active = 0;
|
|
|
|
data->layers[index].active_rnd = 0;
|
|
|
|
data->layers[index].active_clone = 0;
|
|
|
|
data->layers[index].active_mask = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
customData_update_offsets(data);
|
|
|
|
|
|
|
|
return &data->layers[index];
|
2006-08-28 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
2018-05-30 10:28:24 +02:00
|
|
|
void *CustomData_add_layer(
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomData *data, int type, eCDAllocType alloctype, void *layerdata, int totelem)
|
2006-08-28 01:12:36 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomDataLayer *layer;
|
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
layer = customData_add_layer__internal(
|
|
|
|
data, type, alloctype, layerdata, totelem, typeInfo->defaultname);
|
|
|
|
CustomData_update_typemap(data);
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return layer->data;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
2006-08-28 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
2006-12-24 11:15:54 +00:00
|
|
|
/*same as above but accepts a name*/
|
2019-04-17 06:17:24 +02:00
|
|
|
void *CustomData_add_layer_named(CustomData *data,
|
|
|
|
int type,
|
|
|
|
eCDAllocType alloctype,
|
|
|
|
void *layerdata,
|
|
|
|
int totelem,
|
|
|
|
const char *name)
|
2006-12-24 11:15:54 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomDataLayer *layer;
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
layer = customData_add_layer__internal(data, type, alloctype, layerdata, totelem, name);
|
|
|
|
CustomData_update_typemap(data);
|
2006-12-24 11:15:54 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return layer->data;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2006-12-24 11:15:54 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
2006-12-24 11:15:54 +00:00
|
|
|
}
|
|
|
|
|
2013-05-08 12:57:18 +00:00
|
|
|
bool CustomData_free_layer(CustomData *data, int type, int totelem, int index)
|
2006-11-11 16:38:37 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const int index_first = CustomData_get_layer_index(data, type);
|
|
|
|
const int n = index - index_first;
|
|
|
|
int i;
|
2006-11-11 16:38:37 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
BLI_assert(index >= index_first);
|
|
|
|
if ((index_first == -1) || (n < 0)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
BLI_assert(data->layers[index].type == type);
|
2017-05-27 14:03:15 +10:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
customData_free_layer__internal(&data->layers[index], totelem);
|
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
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = index + 1; i < data->totlayer; ++i) {
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[i - 1] = data->layers[i];
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2006-11-11 16:38:37 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
data->totlayer--;
|
2006-11-11 16:38:37 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* if layer was last of type in array, set new active layer */
|
|
|
|
i = CustomData_get_layer_index__notypemap(data, type);
|
2013-07-15 06:29:09 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (i != -1) {
|
|
|
|
/* don't decrement zero index */
|
|
|
|
const int index_nonzero = n ? n : 1;
|
|
|
|
CustomDataLayer *layer;
|
2013-07-15 09:03:28 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (layer = &data->layers[i]; i < data->totlayer && layer->type == type; i++, layer++) {
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer->active >= index_nonzero) {
|
2019-04-17 06:17:24 +02:00
|
|
|
layer->active--;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (layer->active_rnd >= index_nonzero) {
|
2019-04-17 06:17:24 +02:00
|
|
|
layer->active_rnd--;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (layer->active_clone >= index_nonzero) {
|
2019-04-17 06:17:24 +02:00
|
|
|
layer->active_clone--;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (layer->active_mask >= index_nonzero) {
|
2019-04-17 06:17:24 +02:00
|
|
|
layer->active_mask--;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
}
|
|
|
|
}
|
2006-12-12 21:29:09 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (data->totlayer <= data->maxlayer - CUSTOMDATA_GROW) {
|
2019-04-17 06:17:24 +02:00
|
|
|
customData_resize(data, -CUSTOMDATA_GROW);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2006-11-11 16:38:37 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
customData_update_offsets(data);
|
2006-11-11 16:38:37 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return true;
|
2006-11-11 16:38:37 +00:00
|
|
|
}
|
|
|
|
|
2013-05-08 12:57:18 +00:00
|
|
|
bool CustomData_free_layer_active(CustomData *data, int type, int totelem)
|
2006-12-23 17:07:02 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int index = 0;
|
|
|
|
index = CustomData_get_active_layer_index(data, type);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
return CustomData_free_layer(data, type, totelem, index);
|
2006-12-23 17:07:02 +00:00
|
|
|
}
|
|
|
|
|
2006-12-12 21:29:09 +00:00
|
|
|
void CustomData_free_layers(CustomData *data, int type, int totelem)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const int index = CustomData_get_layer_index(data, type);
|
|
|
|
while (CustomData_free_layer(data, type, totelem, index)) {
|
|
|
|
/* pass */
|
|
|
|
}
|
2006-12-12 21:29:09 +00:00
|
|
|
}
|
|
|
|
|
2013-05-08 12:57:18 +00:00
|
|
|
bool CustomData_has_layer(const CustomData *data, int type)
|
2006-11-11 16:38:37 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
return (CustomData_get_layer_index(data, type) != -1);
|
2006-12-12 21:29:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int CustomData_number_of_layers(const CustomData *data, int type)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i, number = 0;
|
2006-12-12 21:29:09 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < data->totlayer; i++) {
|
|
|
|
if (data->layers[i].type == type) {
|
2019-04-17 06:17:24 +02:00
|
|
|
number++;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
}
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return number;
|
2006-11-11 16:38:37 +00:00
|
|
|
}
|
|
|
|
|
2013-05-09 10:41:05 +00:00
|
|
|
int CustomData_number_of_layers_typemask(const CustomData *data, CustomDataMask mask)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i, number = 0;
|
2013-05-09 10:41:05 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < data->totlayer; i++) {
|
|
|
|
if (mask & CD_TYPE_AS_MASK(data->layers[i].type)) {
|
2019-04-17 06:17:24 +02:00
|
|
|
number++;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
}
|
2013-05-09 10:41:05 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return number;
|
2013-05-09 10:41:05 +00:00
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
static void *customData_duplicate_referenced_layer_index(CustomData *data,
|
|
|
|
const int layer_index,
|
|
|
|
const int totelem)
|
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
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomDataLayer *layer;
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (layer_index == -1) {
|
|
|
|
return NULL;
|
|
|
|
}
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
layer = &data->layers[layer_index];
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (layer->flag & CD_FLAG_NOFREE) {
|
|
|
|
/* MEM_dupallocN won't work in case of complex layers, like e.g.
|
|
|
|
* CD_MDEFORMVERT, which has pointers to allocated data...
|
|
|
|
* So in case a custom copy function is defined, use it!
|
|
|
|
*/
|
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(layer->type);
|
2011-12-19 08:26:53 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (typeInfo->copy) {
|
|
|
|
void *dst_data = MEM_malloc_arrayN(
|
|
|
|
(size_t)totelem, typeInfo->size, "CD duplicate ref layer");
|
|
|
|
typeInfo->copy(layer->data, dst_data, totelem);
|
|
|
|
layer->data = dst_data;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
layer->data = MEM_dupallocN(layer->data);
|
|
|
|
}
|
2011-12-19 08:26:53 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
layer->flag &= ~CD_FLAG_NOFREE;
|
|
|
|
}
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return layer->data;
|
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
|
|
|
}
|
|
|
|
|
2014-11-04 10:06:08 +01:00
|
|
|
void *CustomData_duplicate_referenced_layer(CustomData *data, const int type, const int totelem)
|
2007-01-29 15:10:55 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int layer_index;
|
2007-01-29 15:10:55 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the layer index of the first layer of type */
|
|
|
|
layer_index = CustomData_get_active_layer_index(data, type);
|
2007-01-29 15:10:55 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return customData_duplicate_referenced_layer_index(data, layer_index, totelem);
|
2014-11-04 10:06:08 +01:00
|
|
|
}
|
2007-01-29 15:10:55 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
void *CustomData_duplicate_referenced_layer_n(CustomData *data,
|
|
|
|
const int type,
|
|
|
|
const int n,
|
|
|
|
const int totelem)
|
2014-11-04 10:06:08 +01:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int layer_index;
|
2011-12-19 08:26:53 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the layer index of the desired layer */
|
|
|
|
layer_index = CustomData_get_layer_index_n(data, type, n);
|
2011-12-19 08:26:53 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return customData_duplicate_referenced_layer_index(data, layer_index, totelem);
|
2014-11-04 10:06:08 +01:00
|
|
|
}
|
2007-01-29 15:10:55 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
void *CustomData_duplicate_referenced_layer_named(CustomData *data,
|
|
|
|
const int type,
|
|
|
|
const char *name,
|
|
|
|
const int totelem)
|
2014-11-04 10:06:08 +01:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int layer_index;
|
2014-11-04 10:06:08 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the layer index of the desired layer */
|
|
|
|
layer_index = CustomData_get_named_layer_index(data, type, name);
|
2014-11-04 10:06:08 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return customData_duplicate_referenced_layer_index(data, layer_index, totelem);
|
2007-01-29 15:10:55 +00:00
|
|
|
}
|
|
|
|
|
2013-05-08 12:57:18 +00:00
|
|
|
bool CustomData_is_referenced_layer(struct CustomData *data, int type)
|
2011-12-06 22:55:41 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomDataLayer *layer;
|
|
|
|
int layer_index;
|
2011-12-06 22:55:41 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the layer index of the first layer of type */
|
|
|
|
layer_index = CustomData_get_active_layer_index(data, type);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer_index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2011-12-06 22:55:41 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
layer = &data->layers[layer_index];
|
2011-12-06 22:55:41 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return (layer->flag & CD_FLAG_NOFREE) != 0;
|
2011-12-06 22:55:41 +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
|
|
|
void CustomData_free_temporary(CustomData *data, int totelem)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomDataLayer *layer;
|
|
|
|
int i, j;
|
|
|
|
bool changed = false;
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0, j = 0; i < data->totlayer; ++i) {
|
|
|
|
layer = &data->layers[i];
|
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
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (i != j) {
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[j] = data->layers[i];
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2011-05-09 04:06:48 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if ((layer->flag & CD_FLAG_TEMPORARY) == CD_FLAG_TEMPORARY) {
|
|
|
|
customData_free_layer__internal(layer, totelem);
|
|
|
|
changed = true;
|
|
|
|
}
|
2019-04-22 09:39:35 +10:00
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
j++;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02: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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
data->totlayer = j;
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (data->totlayer <= data->maxlayer - CUSTOMDATA_GROW) {
|
|
|
|
customData_resize(data, -CUSTOMDATA_GROW);
|
|
|
|
changed = true;
|
|
|
|
}
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (changed) {
|
|
|
|
customData_update_offsets(data);
|
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
void CustomData_set_only_copy(const struct CustomData *data, CustomDataMask mask)
|
2006-12-05 17:42:03 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2006-12-05 17:42:03 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (!(mask & CD_TYPE_AS_MASK(data->layers[i].type))) {
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[i].flag |= CD_FLAG_NOCOPY;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
}
|
2006-12-05 17:42:03 +00:00
|
|
|
}
|
|
|
|
|
2015-02-23 13:43:09 +11:00
|
|
|
void CustomData_copy_elements(int type, void *src_data_ofs, void *dst_data_ofs, int count)
|
2009-08-18 20:05:08 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
2009-08-18 20:05:08 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->copy) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->copy(src_data_ofs, dst_data_ofs, count);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(dst_data_ofs, src_data_ofs, (size_t)count * typeInfo->size);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2009-08-18 20:05:08 +00:00
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
static void CustomData_copy_data_layer(const CustomData *source,
|
|
|
|
CustomData *dest,
|
|
|
|
int src_i,
|
|
|
|
int dst_i,
|
|
|
|
int src_index,
|
|
|
|
int dst_index,
|
|
|
|
int count)
|
2015-08-06 12:34:31 +10:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo;
|
2013-09-21 05:42:34 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
const void *src_data = source->layers[src_i].data;
|
|
|
|
void *dst_data = dest->layers[dst_i].data;
|
2013-09-21 05:42:34 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo = layerType_getInfo(source->layers[src_i].type);
|
2013-09-21 05:42:34 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
const size_t src_offset = (size_t)src_index * typeInfo->size;
|
|
|
|
const size_t dst_offset = (size_t)dst_index * typeInfo->size;
|
2013-09-21 05:42:34 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (!count || !src_data || !dst_data) {
|
|
|
|
if (count && !(src_data == NULL && dst_data == NULL)) {
|
|
|
|
CLOG_WARN(&LOG,
|
|
|
|
"null data for %s type (%p --> %p), skipping",
|
|
|
|
layerType_getName(source->layers[src_i].type),
|
|
|
|
(void *)src_data,
|
|
|
|
(void *)dst_data);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2013-09-21 05:42:34 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (typeInfo->copy) {
|
|
|
|
typeInfo->copy(
|
|
|
|
POINTER_OFFSET(src_data, src_offset), POINTER_OFFSET(dst_data, dst_offset), count);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
memcpy(POINTER_OFFSET(dst_data, dst_offset),
|
|
|
|
POINTER_OFFSET(src_data, src_offset),
|
|
|
|
(size_t)count * typeInfo->size);
|
|
|
|
}
|
2013-09-21 05:42:34 +00:00
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
void CustomData_copy_data_named(
|
|
|
|
const CustomData *source, CustomData *dest, int source_index, int dest_index, int count)
|
2013-09-21 05:42:34 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int src_i, dest_i;
|
2013-09-21 05:42:34 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* copies a layer at a time */
|
|
|
|
for (src_i = 0; src_i < source->totlayer; ++src_i) {
|
2013-09-21 05:42:34 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
dest_i = CustomData_get_named_layer_index(
|
|
|
|
dest, source->layers[src_i].type, source->layers[src_i].name);
|
2013-09-21 05:42:34 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* if we found a matching layer, copy the data */
|
|
|
|
if (dest_i != -1) {
|
|
|
|
CustomData_copy_data_layer(source, dest, src_i, dest_i, source_index, dest_index, count);
|
|
|
|
}
|
|
|
|
}
|
2013-09-21 05:42:34 +00:00
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
void CustomData_copy_data(
|
|
|
|
const CustomData *source, CustomData *dest, int source_index, int dest_index, int count)
|
2006-08-28 01:12:36 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int src_i, dest_i;
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* copies a layer at a time */
|
|
|
|
dest_i = 0;
|
|
|
|
for (src_i = 0; src_i < source->totlayer; ++src_i) {
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* find the first dest layer with type >= the source type
|
|
|
|
* (this should work because layers are ordered by type)
|
|
|
|
*/
|
|
|
|
while (dest_i < dest->totlayer && dest->layers[dest_i].type < source->layers[src_i].type) {
|
|
|
|
dest_i++;
|
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* if there are no more dest layers, we're done */
|
2019-04-22 09:39:35 +10:00
|
|
|
if (dest_i >= dest->totlayer) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* if we found a matching layer, copy the data */
|
|
|
|
if (dest->layers[dest_i].type == source->layers[src_i].type) {
|
|
|
|
CustomData_copy_data_layer(source, dest, src_i, dest_i, source_index, dest_index, count);
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* if there are multiple source & dest layers of the same type,
|
|
|
|
* we don't want to copy all source layers to the same dest, so
|
|
|
|
* increment dest_i
|
|
|
|
*/
|
|
|
|
dest_i++;
|
|
|
|
}
|
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
2018-09-25 12:27:43 +02:00
|
|
|
void CustomData_copy_layer_type_data(const CustomData *source,
|
|
|
|
CustomData *destination,
|
|
|
|
int type,
|
2019-04-17 06:17:24 +02:00
|
|
|
int source_index,
|
|
|
|
int destination_index,
|
2018-09-25 12:27:43 +02:00
|
|
|
int count)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const int source_layer_index = CustomData_get_layer_index(source, type);
|
|
|
|
if (source_layer_index == -1) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const int destinaiton_layer_index = CustomData_get_layer_index(destination, type);
|
|
|
|
if (destinaiton_layer_index == -1) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
CustomData_copy_data_layer(source,
|
|
|
|
destination,
|
|
|
|
source_layer_index,
|
|
|
|
destinaiton_layer_index,
|
|
|
|
source_index,
|
|
|
|
destination_index,
|
|
|
|
count);
|
2018-09-25 12:27:43 +02: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
|
|
|
void CustomData_free_elem(CustomData *data, int index, int count)
|
2006-08-28 01:12:36 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
|
|
|
const LayerTypeInfo *typeInfo;
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (!(data->layers[i].flag & CD_FLAG_NOFREE)) {
|
|
|
|
typeInfo = layerType_getInfo(data->layers[i].type);
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (typeInfo->free) {
|
|
|
|
size_t offset = (size_t)index * typeInfo->size;
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->free(POINTER_OFFSET(data->layers[i].data, offset), count, typeInfo->size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#define SOURCE_BUF_SIZE 100
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
void CustomData_interp(const CustomData *source,
|
|
|
|
CustomData *dest,
|
|
|
|
const int *src_indices,
|
|
|
|
const float *weights,
|
|
|
|
const float *sub_weights,
|
|
|
|
int count,
|
|
|
|
int dest_index)
|
|
|
|
{
|
|
|
|
int src_i, dest_i;
|
|
|
|
int j;
|
|
|
|
const void *source_buf[SOURCE_BUF_SIZE];
|
|
|
|
const void **sources = source_buf;
|
|
|
|
|
|
|
|
/* slow fallback in case we're interpolating a ridiculous number of
|
|
|
|
* elements
|
|
|
|
*/
|
2019-04-22 09:39:35 +10:00
|
|
|
if (count > SOURCE_BUF_SIZE) {
|
2019-04-17 06:17:24 +02:00
|
|
|
sources = MEM_malloc_arrayN(count, sizeof(*sources), __func__);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
/* interpolates a layer at a time */
|
|
|
|
dest_i = 0;
|
|
|
|
for (src_i = 0; src_i < source->totlayer; ++src_i) {
|
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(source->layers[src_i].type);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!typeInfo->interp) {
|
2019-04-17 06:17:24 +02:00
|
|
|
continue;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
/* find the first dest layer with type >= the source type
|
|
|
|
* (this should work because layers are ordered by type)
|
|
|
|
*/
|
|
|
|
while (dest_i < dest->totlayer && dest->layers[dest_i].type < source->layers[src_i].type) {
|
|
|
|
dest_i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if there are no more dest layers, we're done */
|
2019-04-22 09:39:35 +10:00
|
|
|
if (dest_i >= dest->totlayer) {
|
2019-04-17 06:17:24 +02:00
|
|
|
break;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
/* if we found a matching layer, copy the data */
|
|
|
|
if (dest->layers[dest_i].type == source->layers[src_i].type) {
|
|
|
|
void *src_data = source->layers[src_i].data;
|
|
|
|
|
|
|
|
for (j = 0; j < count; ++j) {
|
|
|
|
sources[j] = POINTER_OFFSET(src_data, (size_t)src_indices[j] * typeInfo->size);
|
|
|
|
}
|
|
|
|
|
|
|
|
typeInfo->interp(
|
|
|
|
sources,
|
|
|
|
weights,
|
|
|
|
sub_weights,
|
|
|
|
count,
|
|
|
|
POINTER_OFFSET(dest->layers[dest_i].data, (size_t)dest_index * typeInfo->size));
|
|
|
|
|
|
|
|
/* if there are multiple source & dest layers of the same type,
|
|
|
|
* we don't want to copy all source layers to the same dest, so
|
|
|
|
* increment dest_i
|
|
|
|
*/
|
|
|
|
dest_i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (count > SOURCE_BUF_SIZE) {
|
2019-04-17 06:17:24 +02:00
|
|
|
MEM_freeN((void *)sources);
|
2019-04-22 09:39:35 +10: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
|
|
|
}
|
|
|
|
|
2016-02-28 15:16:42 +01:00
|
|
|
/**
|
|
|
|
* Swap data inside each item, for all layers.
|
2019-04-27 12:07:07 +10:00
|
|
|
* This only applies to item types that may store several sub-item data
|
|
|
|
* (e.g. corner data [UVs, VCol, ...] of tessellated faces).
|
2016-02-28 15:16:42 +01:00
|
|
|
*
|
2018-12-12 12:50:58 +11:00
|
|
|
* \param corner_indices: A mapping 'new_index -> old_index' of sub-item data.
|
2016-02-28 15:16:42 +01:00
|
|
|
*/
|
|
|
|
void CustomData_swap_corners(struct CustomData *data, int index, const int *corner_indices)
|
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
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo;
|
|
|
|
int i;
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
typeInfo = layerType_getInfo(data->layers[i].type);
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (typeInfo->swap) {
|
|
|
|
const size_t offset = (size_t)index * typeInfo->size;
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->swap(POINTER_OFFSET(data->layers[i].data, offset), corner_indices);
|
|
|
|
}
|
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
2016-02-28 15:29:56 +01:00
|
|
|
/**
|
|
|
|
* Swap two items of given custom data, in all available layers.
|
|
|
|
*/
|
|
|
|
void CustomData_swap(struct CustomData *data, const int index_a, const int index_b)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
|
|
|
char buff_static[256];
|
2016-02-28 15:29:56 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (index_a == index_b) {
|
|
|
|
return;
|
|
|
|
}
|
2016-02-28 15:29:56 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(data->layers[i].type);
|
|
|
|
const size_t size = typeInfo->size;
|
|
|
|
const size_t offset_a = size * index_a;
|
|
|
|
const size_t offset_b = size * index_b;
|
2016-02-28 15:29:56 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
void *buff = size <= sizeof(buff_static) ? buff_static : MEM_mallocN(size, __func__);
|
|
|
|
memcpy(buff, POINTER_OFFSET(data->layers[i].data, offset_a), size);
|
|
|
|
memcpy(POINTER_OFFSET(data->layers[i].data, offset_a),
|
|
|
|
POINTER_OFFSET(data->layers[i].data, offset_b),
|
|
|
|
size);
|
|
|
|
memcpy(POINTER_OFFSET(data->layers[i].data, offset_b), buff, size);
|
2016-02-28 15:29:56 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (buff != buff_static) {
|
|
|
|
MEM_freeN(buff);
|
|
|
|
}
|
|
|
|
}
|
2016-02-28 15:29:56 +01:00
|
|
|
}
|
|
|
|
|
2006-08-28 01:12:36 +00:00
|
|
|
void *CustomData_get(const CustomData *data, int index, int type)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int layer_index;
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
BLI_assert(index >= 0);
|
2013-07-10 05:38:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the layer index of the active layer of type */
|
|
|
|
layer_index = CustomData_get_active_layer_index(data, type);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer_index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the offset of the desired element */
|
|
|
|
const size_t offset = (size_t)index * layerType_getInfo(type)->size;
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return POINTER_OFFSET(data->layers[layer_index].data, offset);
|
2006-08-28 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
2009-05-16 16:18:08 +00:00
|
|
|
void *CustomData_get_n(const CustomData *data, int type, int index, int n)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int layer_index;
|
2009-05-16 16:18:08 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
BLI_assert(index >= 0 && n >= 0);
|
2013-07-10 05:38:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the layer index of the first layer of type */
|
|
|
|
layer_index = data->typemap[type];
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer_index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2011-11-20 16:19:56 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
const size_t offset = (size_t)index * layerType_getInfo(type)->size;
|
|
|
|
return POINTER_OFFSET(data->layers[layer_index + n].data, offset);
|
2009-05-16 16:18:08 +00:00
|
|
|
}
|
|
|
|
|
2006-08-28 01:12:36 +00:00
|
|
|
void *CustomData_get_layer(const CustomData *data, int type)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the layer index of the active layer of type */
|
|
|
|
int layer_index = CustomData_get_active_layer_index(data, type);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer_index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return data->layers[layer_index].data;
|
2006-08-28 01:12:36 +00:00
|
|
|
}
|
|
|
|
|
2006-12-12 21:29:09 +00:00
|
|
|
void *CustomData_get_layer_n(const CustomData *data, int type, int n)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the layer index of the active layer of type */
|
|
|
|
int layer_index = CustomData_get_layer_index_n(data, type, n);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer_index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2006-12-12 21:29:09 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return data->layers[layer_index].data;
|
2006-12-12 21:29:09 +00:00
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
void *CustomData_get_layer_named(const struct CustomData *data, int type, const char *name)
|
2007-01-29 15:10:55 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int layer_index = CustomData_get_named_layer_index(data, type, name);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer_index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2007-01-29 15:10:55 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return data->layers[layer_index].data;
|
2007-01-29 15:10:55 +00:00
|
|
|
}
|
|
|
|
|
2013-01-10 04:43:31 +00:00
|
|
|
int CustomData_get_offset(const CustomData *data, int type)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the layer index of the active layer of type */
|
|
|
|
int layer_index = CustomData_get_active_layer_index(data, type);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer_index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return -1;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2013-01-10 04:43:31 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return data->layers[layer_index].offset;
|
2013-01-10 04:43:31 +00:00
|
|
|
}
|
2011-05-10 17:01:26 +00:00
|
|
|
|
2013-01-10 12:07:01 +00:00
|
|
|
int CustomData_get_n_offset(const CustomData *data, int type, int n)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the layer index of the active layer of type */
|
|
|
|
int layer_index = CustomData_get_layer_index_n(data, type, n);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer_index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return -1;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2013-01-10 12:07:01 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return data->layers[layer_index].offset;
|
2013-01-10 12:07:01 +00:00
|
|
|
}
|
|
|
|
|
2013-05-08 12:57:18 +00:00
|
|
|
bool CustomData_set_layer_name(const CustomData *data, int type, int n, const char *name)
|
2011-05-10 17:01:26 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the layer index of the first layer of type */
|
|
|
|
const int layer_index = CustomData_get_layer_index_n(data, type, n);
|
2011-05-10 17:01:26 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if ((layer_index == -1) || !name) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
BLI_strncpy(data->layers[layer_index].name, name, sizeof(data->layers[layer_index].name));
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return true;
|
2011-05-10 17:01:26 +00:00
|
|
|
}
|
|
|
|
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
const char *CustomData_get_layer_name(const CustomData *data, int type, int n)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const int layer_index = CustomData_get_layer_index_n(data, type, n);
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return (layer_index == -1) ? NULL : data->layers[layer_index].name;
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01: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
|
|
|
void *CustomData_set_layer(const CustomData *data, int type, void *ptr)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the layer index of the first layer of type */
|
|
|
|
int layer_index = CustomData_get_active_layer_index(data, type);
|
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
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer_index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
2019-04-22 09:39:35 +10: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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[layer_index].data = ptr;
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return ptr;
|
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
|
|
|
}
|
|
|
|
|
2007-01-10 06:09:10 +00:00
|
|
|
void *CustomData_set_layer_n(const struct CustomData *data, int type, int n, void *ptr)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the layer index of the first layer of type */
|
|
|
|
int layer_index = CustomData_get_layer_index_n(data, type, n);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer_index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2007-01-10 06:09:10 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[layer_index].data = ptr;
|
2007-01-10 06:09:10 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return ptr;
|
2007-01-10 06:09:10 +00:00
|
|
|
}
|
|
|
|
|
2015-02-23 15:55:48 +11:00
|
|
|
void CustomData_set(const CustomData *data, int index, int type, const void *source)
|
2006-08-28 01:12:36 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
void *dest = CustomData_get(data, index, type);
|
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!dest) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->copy) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->copy(source, dest, 1);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(dest, source, typeInfo->size);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2006-08-28 01:12:36 +00:00
|
|
|
}
|
2006-11-11 16:38:37 +00:00
|
|
|
|
2012-08-17 14:43:20 +00:00
|
|
|
/* BMesh functions */
|
|
|
|
/* needed to convert to/from different face reps */
|
2017-05-25 15:11:00 +10:00
|
|
|
void CustomData_to_bmeshpoly(CustomData *fdata, CustomData *ldata, int totloop)
|
2008-07-08 02:22:37 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
for (int i = 0; i < fdata->totlayer; i++) {
|
|
|
|
if (fdata->layers[i].type == CD_MTFACE) {
|
|
|
|
CustomData_add_layer_named(
|
|
|
|
ldata, CD_MLOOPUV, CD_CALLOC, NULL, totloop, fdata->layers[i].name);
|
|
|
|
}
|
|
|
|
else if (fdata->layers[i].type == CD_MCOL) {
|
|
|
|
CustomData_add_layer_named(
|
|
|
|
ldata, CD_MLOOPCOL, CD_CALLOC, NULL, totloop, fdata->layers[i].name);
|
|
|
|
}
|
|
|
|
else if (fdata->layers[i].type == CD_MDISPS) {
|
|
|
|
CustomData_add_layer_named(
|
|
|
|
ldata, CD_MDISPS, CD_CALLOC, NULL, totloop, fdata->layers[i].name);
|
|
|
|
}
|
|
|
|
else if (fdata->layers[i].type == CD_TESSLOOPNORMAL) {
|
|
|
|
CustomData_add_layer_named(
|
|
|
|
ldata, CD_NORMAL, CD_CALLOC, NULL, totloop, fdata->layers[i].name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CustomData_from_bmeshpoly(CustomData *fdata, CustomData *ldata, int total)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* avoid accumulating extra layers */
|
|
|
|
BLI_assert(!CustomData_from_bmeshpoly_test(fdata, ldata, false));
|
|
|
|
|
|
|
|
for (i = 0; i < ldata->totlayer; i++) {
|
|
|
|
if (ldata->layers[i].type == CD_MLOOPUV) {
|
|
|
|
CustomData_add_layer_named(fdata, CD_MTFACE, CD_CALLOC, NULL, total, ldata->layers[i].name);
|
|
|
|
}
|
|
|
|
if (ldata->layers[i].type == CD_MLOOPCOL) {
|
|
|
|
CustomData_add_layer_named(fdata, CD_MCOL, CD_CALLOC, NULL, total, ldata->layers[i].name);
|
|
|
|
}
|
|
|
|
else if (ldata->layers[i].type == CD_PREVIEW_MLOOPCOL) {
|
|
|
|
CustomData_add_layer_named(
|
|
|
|
fdata, CD_PREVIEW_MCOL, CD_CALLOC, NULL, total, ldata->layers[i].name);
|
|
|
|
}
|
|
|
|
else if (ldata->layers[i].type == CD_ORIGSPACE_MLOOP) {
|
|
|
|
CustomData_add_layer_named(
|
|
|
|
fdata, CD_ORIGSPACE, CD_CALLOC, NULL, total, ldata->layers[i].name);
|
|
|
|
}
|
|
|
|
else if (ldata->layers[i].type == CD_NORMAL) {
|
|
|
|
CustomData_add_layer_named(
|
|
|
|
fdata, CD_TESSLOOPNORMAL, CD_CALLOC, NULL, total, ldata->layers[i].name);
|
|
|
|
}
|
|
|
|
else if (ldata->layers[i].type == CD_TANGENT) {
|
|
|
|
CustomData_add_layer_named(fdata, CD_TANGENT, CD_CALLOC, NULL, total, ldata->layers[i].name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CustomData_bmesh_update_active_layers(fdata, ldata);
|
2008-07-08 02:22:37 +00:00
|
|
|
}
|
|
|
|
|
2015-09-09 16:42:55 +10:00
|
|
|
#ifndef NDEBUG
|
|
|
|
/**
|
|
|
|
* Debug check, used to assert when we expect layers to be in/out of sync.
|
|
|
|
*
|
|
|
|
* \param fallback: Use when there are no layers to handle,
|
|
|
|
* since callers may expect success or failure.
|
|
|
|
*/
|
2017-05-25 15:11:00 +10:00
|
|
|
bool CustomData_from_bmeshpoly_test(CustomData *fdata, CustomData *ldata, bool fallback)
|
2015-09-09 16:42:55 +10:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int a_num = 0, b_num = 0;
|
|
|
|
# define LAYER_CMP(l_a, t_a, l_b, t_b) \
|
|
|
|
((a_num += CustomData_number_of_layers(l_a, t_a)) == \
|
|
|
|
(b_num += CustomData_number_of_layers(l_b, t_b)))
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!LAYER_CMP(ldata, CD_MLOOPUV, fdata, CD_MTFACE)) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (!LAYER_CMP(ldata, CD_MLOOPCOL, fdata, CD_MCOL)) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (!LAYER_CMP(ldata, CD_PREVIEW_MLOOPCOL, fdata, CD_PREVIEW_MCOL)) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (!LAYER_CMP(ldata, CD_ORIGSPACE_MLOOP, fdata, CD_ORIGSPACE)) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (!LAYER_CMP(ldata, CD_NORMAL, fdata, CD_TESSLOOPNORMAL)) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (!LAYER_CMP(ldata, CD_TANGENT, fdata, CD_TANGENT)) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
# undef LAYER_CMP
|
|
|
|
|
|
|
|
/* if no layers are on either CustomData's,
|
|
|
|
* then there was nothing to do... */
|
|
|
|
return a_num ? true : fallback;
|
2015-09-09 16:42:55 +10:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-05-25 15:11:00 +10:00
|
|
|
void CustomData_bmesh_update_active_layers(CustomData *fdata, CustomData *ldata)
|
2011-12-01 09:49:27 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int act;
|
2011-12-01 09:49:27 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (CustomData_has_layer(ldata, CD_MLOOPUV)) {
|
|
|
|
act = CustomData_get_active_layer(ldata, CD_MLOOPUV);
|
|
|
|
CustomData_set_layer_active(fdata, CD_MTFACE, act);
|
2011-12-01 09:49:27 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
act = CustomData_get_render_layer(ldata, CD_MLOOPUV);
|
|
|
|
CustomData_set_layer_render(fdata, CD_MTFACE, act);
|
2011-12-01 09:49:27 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
act = CustomData_get_clone_layer(ldata, CD_MLOOPUV);
|
|
|
|
CustomData_set_layer_clone(fdata, CD_MTFACE, act);
|
2011-12-01 09:49:27 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
act = CustomData_get_stencil_layer(ldata, CD_MLOOPUV);
|
|
|
|
CustomData_set_layer_stencil(fdata, CD_MTFACE, act);
|
|
|
|
}
|
2011-12-01 09:49:27 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (CustomData_has_layer(ldata, CD_MLOOPCOL)) {
|
|
|
|
act = CustomData_get_active_layer(ldata, CD_MLOOPCOL);
|
|
|
|
CustomData_set_layer_active(fdata, CD_MCOL, act);
|
2011-12-01 09:49:27 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
act = CustomData_get_render_layer(ldata, CD_MLOOPCOL);
|
|
|
|
CustomData_set_layer_render(fdata, CD_MCOL, act);
|
2011-12-01 09:49:27 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
act = CustomData_get_clone_layer(ldata, CD_MLOOPCOL);
|
|
|
|
CustomData_set_layer_clone(fdata, CD_MCOL, act);
|
2011-12-01 09:49:27 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
act = CustomData_get_stencil_layer(ldata, CD_MLOOPCOL);
|
|
|
|
CustomData_set_layer_stencil(fdata, CD_MCOL, act);
|
|
|
|
}
|
2011-12-01 09:49:27 +00:00
|
|
|
}
|
2008-07-08 02:22:37 +00:00
|
|
|
|
2012-06-07 09:11:16 +00:00
|
|
|
/* update active indices for active/render/clone/stencil custom data layers
|
|
|
|
* based on indices from fdata layers
|
|
|
|
* used by do_versions in readfile.c when creating pdata and ldata for pre-bmesh
|
|
|
|
* meshes and needed to preserve active/render/clone/stencil flags set in pre-bmesh files
|
|
|
|
*/
|
2017-05-25 15:11:00 +10:00
|
|
|
void CustomData_bmesh_do_versions_update_active_layers(CustomData *fdata, CustomData *ldata)
|
2012-06-07 09:11:16 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int act;
|
2012-06-07 09:11:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (CustomData_has_layer(fdata, CD_MTFACE)) {
|
|
|
|
act = CustomData_get_active_layer(fdata, CD_MTFACE);
|
|
|
|
CustomData_set_layer_active(ldata, CD_MLOOPUV, act);
|
2012-06-07 09:11:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
act = CustomData_get_render_layer(fdata, CD_MTFACE);
|
|
|
|
CustomData_set_layer_render(ldata, CD_MLOOPUV, act);
|
2012-06-07 09:11:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
act = CustomData_get_clone_layer(fdata, CD_MTFACE);
|
|
|
|
CustomData_set_layer_clone(ldata, CD_MLOOPUV, act);
|
2012-06-07 09:11:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
act = CustomData_get_stencil_layer(fdata, CD_MTFACE);
|
|
|
|
CustomData_set_layer_stencil(ldata, CD_MLOOPUV, act);
|
|
|
|
}
|
2012-06-07 09:11:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (CustomData_has_layer(fdata, CD_MCOL)) {
|
|
|
|
act = CustomData_get_active_layer(fdata, CD_MCOL);
|
|
|
|
CustomData_set_layer_active(ldata, CD_MLOOPCOL, act);
|
2012-06-07 09:11:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
act = CustomData_get_render_layer(fdata, CD_MCOL);
|
|
|
|
CustomData_set_layer_render(ldata, CD_MLOOPCOL, act);
|
2012-06-07 09:11:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
act = CustomData_get_clone_layer(fdata, CD_MCOL);
|
|
|
|
CustomData_set_layer_clone(ldata, CD_MLOOPCOL, act);
|
2012-06-07 09:11:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
act = CustomData_get_stencil_layer(fdata, CD_MCOL);
|
|
|
|
CustomData_set_layer_stencil(ldata, CD_MLOOPCOL, act);
|
|
|
|
}
|
2012-06-07 09:11:16 +00:00
|
|
|
}
|
|
|
|
|
2012-03-01 22:17:04 +00:00
|
|
|
void CustomData_bmesh_init_pool(CustomData *data, int totelem, const char htype)
|
2011-11-07 09:02:10 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int chunksize;
|
|
|
|
|
|
|
|
/* Dispose old pools before calling here to avoid leaks */
|
|
|
|
BLI_assert(data->pool == NULL);
|
|
|
|
|
|
|
|
switch (htype) {
|
|
|
|
case BM_VERT:
|
|
|
|
chunksize = bm_mesh_chunksize_default.totvert;
|
|
|
|
break;
|
|
|
|
case BM_EDGE:
|
|
|
|
chunksize = bm_mesh_chunksize_default.totedge;
|
|
|
|
break;
|
|
|
|
case BM_LOOP:
|
|
|
|
chunksize = bm_mesh_chunksize_default.totloop;
|
|
|
|
break;
|
|
|
|
case BM_FACE:
|
|
|
|
chunksize = bm_mesh_chunksize_default.totface;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
BLI_assert(0);
|
|
|
|
chunksize = 512;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If there are no layers, no pool is needed just yet */
|
|
|
|
if (data->totlayer) {
|
|
|
|
data->pool = BLI_mempool_create(data->totsize, totelem, chunksize, BLI_MEMPOOL_NOP);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CustomData_bmesh_merge(const CustomData *source,
|
|
|
|
CustomData *dest,
|
|
|
|
CustomDataMask mask,
|
|
|
|
eCDAllocType alloctype,
|
|
|
|
BMesh *bm,
|
|
|
|
const char htype)
|
|
|
|
{
|
|
|
|
BMHeader *h;
|
|
|
|
BMIter iter;
|
|
|
|
CustomData destold;
|
|
|
|
void *tmp;
|
|
|
|
int iter_type;
|
|
|
|
int totelem;
|
|
|
|
|
|
|
|
if (CustomData_number_of_layers_typemask(source, mask) == 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy old layer description so that old data can be copied into
|
|
|
|
* the new allocation */
|
|
|
|
destold = *dest;
|
|
|
|
if (destold.layers) {
|
|
|
|
destold.layers = MEM_dupallocN(destold.layers);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (CustomData_merge(source, dest, mask, alloctype, 0) == false) {
|
2019-04-22 09:39:35 +10:00
|
|
|
if (destold.layers) {
|
2019-04-17 06:17:24 +02:00
|
|
|
MEM_freeN(destold.layers);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (htype) {
|
|
|
|
case BM_VERT:
|
|
|
|
iter_type = BM_VERTS_OF_MESH;
|
|
|
|
totelem = bm->totvert;
|
|
|
|
break;
|
|
|
|
case BM_EDGE:
|
|
|
|
iter_type = BM_EDGES_OF_MESH;
|
|
|
|
totelem = bm->totedge;
|
|
|
|
break;
|
|
|
|
case BM_LOOP:
|
|
|
|
iter_type = BM_LOOPS_OF_FACE;
|
|
|
|
totelem = bm->totloop;
|
|
|
|
break;
|
|
|
|
case BM_FACE:
|
|
|
|
iter_type = BM_FACES_OF_MESH;
|
|
|
|
totelem = bm->totface;
|
|
|
|
break;
|
|
|
|
default: /* should never happen */
|
|
|
|
BLI_assert(!"invalid type given");
|
|
|
|
iter_type = BM_VERTS_OF_MESH;
|
|
|
|
totelem = bm->totvert;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
dest->pool = NULL;
|
|
|
|
CustomData_bmesh_init_pool(dest, totelem, htype);
|
|
|
|
|
|
|
|
if (iter_type != BM_LOOPS_OF_FACE) {
|
|
|
|
/*ensure all current elements follow new customdata layout*/
|
|
|
|
BM_ITER_MESH (h, &iter, bm, iter_type) {
|
|
|
|
tmp = NULL;
|
|
|
|
CustomData_bmesh_copy_data(&destold, dest, h->data, &tmp);
|
|
|
|
CustomData_bmesh_free_block(&destold, &h->data);
|
|
|
|
h->data = tmp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
BMFace *f;
|
|
|
|
BMLoop *l;
|
|
|
|
BMIter liter;
|
|
|
|
|
|
|
|
/*ensure all current elements follow new customdata layout*/
|
|
|
|
BM_ITER_MESH (f, &iter, bm, BM_FACES_OF_MESH) {
|
|
|
|
BM_ITER_ELEM (l, &liter, f, BM_LOOPS_OF_FACE) {
|
|
|
|
tmp = NULL;
|
|
|
|
CustomData_bmesh_copy_data(&destold, dest, l->head.data, &tmp);
|
|
|
|
CustomData_bmesh_free_block(&destold, &l->head.data);
|
|
|
|
l->head.data = tmp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (destold.pool) {
|
2019-04-17 06:17:24 +02:00
|
|
|
BLI_mempool_destroy(destold.pool);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
if (destold.layers) {
|
2019-04-17 06:17:24 +02:00
|
|
|
MEM_freeN(destold.layers);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
return true;
|
2009-06-23 05:35:49 +00:00
|
|
|
}
|
|
|
|
|
2008-07-04 17:59:16 +00:00
|
|
|
void CustomData_bmesh_free_block(CustomData *data, void **block)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo;
|
|
|
|
int i;
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (*block == NULL) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2012-10-10 23:44:07 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (!(data->layers[i].flag & CD_FLAG_NOFREE)) {
|
|
|
|
typeInfo = layerType_getInfo(data->layers[i].type);
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (typeInfo->free) {
|
|
|
|
int offset = data->layers[i].offset;
|
|
|
|
typeInfo->free(POINTER_OFFSET(*block, offset), 1, typeInfo->size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (data->totsize) {
|
2019-04-17 06:17:24 +02:00
|
|
|
BLI_mempool_free(data->pool, *block);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2010-07-14 22:06:10 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
*block = NULL;
|
2008-07-04 17:59:16 +00:00
|
|
|
}
|
|
|
|
|
2013-05-08 13:00:25 +00:00
|
|
|
/**
|
|
|
|
* Same as #CustomData_bmesh_free_block but zero the memory rather then freeing.
|
|
|
|
*/
|
2014-09-24 18:50:29 +10:00
|
|
|
void CustomData_bmesh_free_block_data(CustomData *data, void *block)
|
2013-05-08 13:00:25 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo;
|
|
|
|
int i;
|
2013-05-08 13:00:25 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (block == NULL) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2013-05-08 13:00:25 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (!(data->layers[i].flag & CD_FLAG_NOFREE)) {
|
|
|
|
typeInfo = layerType_getInfo(data->layers[i].type);
|
2013-05-08 13:00:25 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (typeInfo->free) {
|
|
|
|
const size_t offset = data->layers[i].offset;
|
|
|
|
typeInfo->free(POINTER_OFFSET(block, offset), 1, typeInfo->size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-05-08 13:00:25 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (data->totsize) {
|
2019-04-17 06:17:24 +02:00
|
|
|
memset(block, 0, data->totsize);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2013-05-08 13:00:25 +00:00
|
|
|
}
|
|
|
|
|
2008-07-04 17:59:16 +00:00
|
|
|
static void CustomData_bmesh_alloc_block(CustomData *data, void **block)
|
|
|
|
{
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (*block) {
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomData_bmesh_free_block(data, block);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (data->totsize > 0) {
|
2019-04-17 06:17:24 +02:00
|
|
|
*block = BLI_mempool_alloc(data->pool);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
*block = NULL;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2008-07-04 17:59:16 +00:00
|
|
|
}
|
|
|
|
|
2019-06-17 19:05:41 +02:00
|
|
|
static void CustomData_bmesh_set_default_n(CustomData *data, void **block, int n)
|
|
|
|
{
|
|
|
|
const LayerTypeInfo *typeInfo;
|
|
|
|
int offset = data->layers[n].offset;
|
|
|
|
|
|
|
|
typeInfo = layerType_getInfo(data->layers[n].type);
|
|
|
|
|
|
|
|
if (typeInfo->set_default) {
|
|
|
|
typeInfo->set_default(POINTER_OFFSET(*block, offset), 1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
memset(POINTER_OFFSET(*block, offset), 0, typeInfo->size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CustomData_bmesh_set_default(CustomData *data, void **block)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (*block == NULL) {
|
|
|
|
CustomData_bmesh_alloc_block(data, block);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
CustomData_bmesh_set_default_n(data, block, i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
void CustomData_bmesh_copy_data(const CustomData *source,
|
|
|
|
CustomData *dest,
|
|
|
|
void *src_block,
|
|
|
|
void **dest_block)
|
2008-07-04 17:59:16 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo;
|
|
|
|
int dest_i, src_i;
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (*dest_block == NULL) {
|
|
|
|
CustomData_bmesh_alloc_block(dest, dest_block);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (*dest_block) {
|
2019-04-17 06:17:24 +02:00
|
|
|
memset(*dest_block, 0, dest->totsize);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
}
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* copies a layer at a time */
|
|
|
|
dest_i = 0;
|
|
|
|
for (src_i = 0; src_i < source->totlayer; ++src_i) {
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* find the first dest layer with type >= the source type
|
|
|
|
* (this should work because layers are ordered by type)
|
|
|
|
*/
|
|
|
|
while (dest_i < dest->totlayer && dest->layers[dest_i].type < source->layers[src_i].type) {
|
2019-06-17 19:05:41 +02:00
|
|
|
CustomData_bmesh_set_default_n(dest, dest_block, dest_i);
|
2019-04-17 06:17:24 +02:00
|
|
|
dest_i++;
|
|
|
|
}
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* if there are no more dest layers, we're done */
|
2019-04-22 09:39:35 +10:00
|
|
|
if (dest_i >= dest->totlayer) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* if we found a matching layer, copy the data */
|
|
|
|
if (dest->layers[dest_i].type == source->layers[src_i].type &&
|
|
|
|
STREQ(dest->layers[dest_i].name, source->layers[src_i].name)) {
|
|
|
|
const void *src_data = POINTER_OFFSET(src_block, source->layers[src_i].offset);
|
|
|
|
void *dest_data = POINTER_OFFSET(*dest_block, dest->layers[dest_i].offset);
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo = layerType_getInfo(source->layers[src_i].type);
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->copy) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->copy(src_data, dest_data, 1);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(dest_data, src_data, typeInfo->size);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* if there are multiple source & dest layers of the same type,
|
|
|
|
* we don't want to copy all source layers to the same dest, so
|
|
|
|
* increment dest_i
|
|
|
|
*/
|
|
|
|
dest_i++;
|
|
|
|
}
|
|
|
|
}
|
2019-06-17 19:05:41 +02:00
|
|
|
|
|
|
|
while (dest_i < dest->totlayer) {
|
|
|
|
CustomData_bmesh_set_default_n(dest, dest_block, dest_i);
|
|
|
|
dest_i++;
|
|
|
|
}
|
2008-07-04 17:59:16 +00:00
|
|
|
}
|
|
|
|
|
2019-04-27 12:07:07 +10:00
|
|
|
/* BMesh Custom Data Functions.
|
|
|
|
* Should replace edit-mesh ones with these as well, due to more efficient memory alloc.
|
|
|
|
*/
|
2008-07-04 17:59:16 +00:00
|
|
|
void *CustomData_bmesh_get(const CustomData *data, void *block, int type)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int layer_index;
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the layer index of the first layer of type */
|
|
|
|
layer_index = CustomData_get_active_layer_index(data, type);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer_index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return POINTER_OFFSET(block, data->layers[layer_index].offset);
|
2008-07-04 17:59:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void *CustomData_bmesh_get_n(const CustomData *data, void *block, int type, int n)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int layer_index;
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* get the layer index of the first layer of type */
|
|
|
|
layer_index = CustomData_get_layer_index(data, type);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer_index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return POINTER_OFFSET(block, data->layers[layer_index + n].offset);
|
2008-07-04 17:59:16 +00:00
|
|
|
}
|
|
|
|
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
/*gets from the layer at physical index n, note: doesn't check type.*/
|
|
|
|
void *CustomData_bmesh_get_layer_n(const CustomData *data, void *block, int n)
|
|
|
|
{
|
2019-04-22 09:39:35 +10:00
|
|
|
if (n < 0 || n >= data->totlayer) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return NULL;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return POINTER_OFFSET(block, data->layers[n].offset);
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
bool CustomData_layer_has_math(const struct CustomData *data, int layer_n)
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(data->layers[layer_n].type);
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (typeInfo->equal && typeInfo->add && typeInfo->multiply && typeInfo->initminmax &&
|
|
|
|
typeInfo->dominmax) {
|
|
|
|
return true;
|
|
|
|
}
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2012-10-20 17:31:07 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
bool CustomData_layer_has_interp(const struct CustomData *data, int layer_n)
|
2012-10-20 17:39:56 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(data->layers[layer_n].type);
|
2012-10-20 17:39:56 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (typeInfo->interp) {
|
|
|
|
return true;
|
|
|
|
}
|
2012-10-20 17:39:56 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2012-10-20 17:39:56 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
bool CustomData_has_math(const struct CustomData *data)
|
2012-10-20 17:31:07 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2012-10-20 17:31:07 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* interpolates a layer at a time */
|
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (CustomData_layer_has_math(data, i)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
2012-10-20 17:31:07 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
2013-05-08 12:59:56 +00:00
|
|
|
/* a non bmesh version would have to check layer->data */
|
2014-09-09 16:12:07 +10:00
|
|
|
bool CustomData_bmesh_has_free(const struct CustomData *data)
|
2013-05-08 12:59:56 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo;
|
|
|
|
int i;
|
2013-05-08 12:59:56 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (!(data->layers[i].flag & CD_FLAG_NOFREE)) {
|
|
|
|
typeInfo = layerType_getInfo(data->layers[i].type);
|
|
|
|
if (typeInfo->free) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
2013-05-08 12:59:56 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
bool CustomData_has_interp(const struct CustomData *data)
|
2012-10-20 17:39:56 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
2012-10-20 17:39:56 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* interpolates a layer at a time */
|
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (CustomData_layer_has_interp(data, i)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
2012-10-20 17:39:56 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2012-10-20 17:39:56 +00:00
|
|
|
}
|
|
|
|
|
2015-02-19 00:00:23 +05:00
|
|
|
bool CustomData_has_referenced(const struct CustomData *data)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
if (data->layers[i].flag & CD_FLAG_NOFREE) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
2015-02-19 00:00:23 +05:00
|
|
|
}
|
|
|
|
|
2012-03-03 20:19:11 +00:00
|
|
|
/* copies the "value" (e.g. mloopuv uv or mloopcol colors) from one block to
|
|
|
|
* another, while not overwriting anything else (e.g. flags)*/
|
2014-09-09 16:12:07 +10:00
|
|
|
void CustomData_data_copy_value(int type, const void *source, void *dest)
|
2009-08-12 03:51:28 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
2009-08-12 03:51:28 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!dest) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2009-08-12 03:51:28 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->copyvalue) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->copyvalue(source, dest, CDT_MIX_NOMIX, 0.0f);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(dest, source, typeInfo->size);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2009-08-12 03:51:28 +00:00
|
|
|
}
|
|
|
|
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
/* Mixes the "value" (e.g. mloopuv uv or mloopcol colors) from one block into
|
|
|
|
* another, while not overwriting anything else (e.g. flags)*/
|
2019-04-17 06:17:24 +02:00
|
|
|
void CustomData_data_mix_value(
|
|
|
|
int type, const void *source, void *dest, const int mixmode, const float mixfactor)
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!dest) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (typeInfo->copyvalue) {
|
|
|
|
typeInfo->copyvalue(source, dest, mixmode, mixfactor);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Mere copy if no advanced interpolation is supported. */
|
|
|
|
memcpy(dest, source, typeInfo->size);
|
|
|
|
}
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
bool CustomData_data_equals(int type, const void *data1, const void *data2)
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->equal) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return typeInfo->equal(data1, data2);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
return !memcmp(data1, data2, typeInfo->size);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CustomData_data_initminmax(int type, void *min, void *max)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->initminmax) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->initminmax(min, max);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
void CustomData_data_dominmax(int type, const void *data, void *min, void *max)
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->dominmax) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->dominmax(data, min, max);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CustomData_data_multiply(int type, void *data, float fac)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->multiply) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->multiply(data, fac);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-09 16:12:07 +10:00
|
|
|
void CustomData_data_add(int type, void *data1, const void *data2)
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->add) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->add(data1, data2);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
2015-02-23 15:55:48 +11:00
|
|
|
void CustomData_bmesh_set(const CustomData *data, void *block, int type, const void *source)
|
2008-07-04 17:59:16 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
void *dest = CustomData_bmesh_get(data, block, type);
|
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!dest) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->copy) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->copy(source, dest, 1);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(dest, source, typeInfo->size);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2008-07-04 17:59:16 +00:00
|
|
|
}
|
|
|
|
|
2015-02-23 15:55:48 +11:00
|
|
|
void CustomData_bmesh_set_n(CustomData *data, void *block, int type, int n, const void *source)
|
2008-07-04 17:59:16 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
void *dest = CustomData_bmesh_get_n(data, block, type, n);
|
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!dest) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2008-07-04 17:59:16 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->copy) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->copy(source, dest, 1);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(dest, source, typeInfo->size);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2008-07-04 17:59:16 +00:00
|
|
|
}
|
|
|
|
|
2015-02-23 15:55:48 +11:00
|
|
|
void CustomData_bmesh_set_layer_n(CustomData *data, void *block, int n, const void *source)
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
void *dest = CustomData_bmesh_get_layer_n(data, block, n);
|
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(data->layers[n].type);
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!dest) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->copy) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->copy(source, dest, 1);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(dest, source, typeInfo->size);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
Brought Extrude all the way back. The contextual menu works,
as does only edges and individual faces extrude (individual vert
extrude already did).
Note that I need to port this, after we all figure out how to handle
operators with variable transform follow-ons.
I also implemented the merge->collapse function, which is currently
accessable under ctrl->v, Bmesh Test Operator. I still need to
implement the other merge modes, and properly hook everything into
the merge menu tool, which I plan on doing soon (tomorrow hopefully).
The cool thing about the collapse tool, is not only does it handle (all)
UV layers, it handles vcols as well. To do this, I had to add a few math
functions to the customdata API, which seem to be working well.
2009-08-11 07:49:35 +00:00
|
|
|
}
|
|
|
|
|
2013-01-08 16:39:36 +00:00
|
|
|
/**
|
2015-02-23 13:43:09 +11:00
|
|
|
* \note src_blocks_ofs & dst_block_ofs
|
|
|
|
* must be pointers to the data, offset by layer->offset already.
|
2013-01-08 16:39:36 +00:00
|
|
|
*/
|
2019-04-17 06:17:24 +02:00
|
|
|
void CustomData_bmesh_interp_n(CustomData *data,
|
|
|
|
const void **src_blocks_ofs,
|
|
|
|
const float *weights,
|
|
|
|
const float *sub_weights,
|
|
|
|
int count,
|
|
|
|
void *dst_block_ofs,
|
|
|
|
int n)
|
|
|
|
{
|
|
|
|
CustomDataLayer *layer = &data->layers[n];
|
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(layer->type);
|
|
|
|
|
|
|
|
typeInfo->interp(src_blocks_ofs, weights, sub_weights, count, dst_block_ofs);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CustomData_bmesh_interp(CustomData *data,
|
|
|
|
const void **src_blocks,
|
|
|
|
const float *weights,
|
|
|
|
const float *sub_weights,
|
|
|
|
int count,
|
|
|
|
void *dst_block)
|
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
void *source_buf[SOURCE_BUF_SIZE];
|
|
|
|
const void **sources = (const void **)source_buf;
|
|
|
|
|
|
|
|
/* slow fallback in case we're interpolating a ridiculous number of
|
|
|
|
* elements
|
|
|
|
*/
|
2019-04-22 09:39:35 +10:00
|
|
|
if (count > SOURCE_BUF_SIZE) {
|
2019-04-17 06:17:24 +02:00
|
|
|
sources = MEM_malloc_arrayN(count, sizeof(*sources), __func__);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
/* interpolates a layer at a time */
|
|
|
|
for (i = 0; i < data->totlayer; ++i) {
|
|
|
|
CustomDataLayer *layer = &data->layers[i];
|
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(layer->type);
|
|
|
|
if (typeInfo->interp) {
|
|
|
|
for (j = 0; j < count; ++j) {
|
|
|
|
sources[j] = POINTER_OFFSET(src_blocks[j], layer->offset);
|
|
|
|
}
|
|
|
|
CustomData_bmesh_interp_n(
|
|
|
|
data, sources, weights, sub_weights, count, POINTER_OFFSET(dst_block, layer->offset), i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (count > SOURCE_BUF_SIZE) {
|
2019-04-17 06:17:24 +02:00
|
|
|
MEM_freeN((void *)sources);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2008-07-04 17:59:16 +00:00
|
|
|
}
|
|
|
|
|
2013-01-08 14:25:17 +00:00
|
|
|
/**
|
2018-12-12 12:50:58 +11:00
|
|
|
* \param use_default_init: initializes data which can't be copied,
|
2013-01-08 14:25:17 +00:00
|
|
|
* typically you'll want to use this if the BM_xxx create function
|
|
|
|
* is called with BM_CREATE_SKIP_CD flag
|
|
|
|
*/
|
2019-04-17 06:17:24 +02:00
|
|
|
void CustomData_to_bmesh_block(const CustomData *source,
|
|
|
|
CustomData *dest,
|
|
|
|
int src_index,
|
|
|
|
void **dest_block,
|
|
|
|
bool use_default_init)
|
|
|
|
{
|
|
|
|
const LayerTypeInfo *typeInfo;
|
|
|
|
int dest_i, src_i;
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (*dest_block == NULL) {
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomData_bmesh_alloc_block(dest, dest_block);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
/* copies a layer at a time */
|
|
|
|
dest_i = 0;
|
|
|
|
for (src_i = 0; src_i < source->totlayer; ++src_i) {
|
|
|
|
|
|
|
|
/* find the first dest layer with type >= the source type
|
|
|
|
* (this should work because layers are ordered by type)
|
|
|
|
*/
|
|
|
|
while (dest_i < dest->totlayer && dest->layers[dest_i].type < source->layers[src_i].type) {
|
|
|
|
if (use_default_init) {
|
|
|
|
CustomData_bmesh_set_default_n(dest, dest_block, dest_i);
|
|
|
|
}
|
|
|
|
dest_i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if there are no more dest layers, we're done */
|
2019-04-22 09:39:35 +10:00
|
|
|
if (dest_i >= dest->totlayer) {
|
2019-04-17 06:17:24 +02:00
|
|
|
break;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
/* if we found a matching layer, copy the data */
|
|
|
|
if (dest->layers[dest_i].type == source->layers[src_i].type) {
|
|
|
|
int offset = dest->layers[dest_i].offset;
|
|
|
|
const void *src_data = source->layers[src_i].data;
|
|
|
|
void *dest_data = POINTER_OFFSET(*dest_block, offset);
|
|
|
|
|
|
|
|
typeInfo = layerType_getInfo(dest->layers[dest_i].type);
|
|
|
|
const size_t src_offset = (size_t)src_index * typeInfo->size;
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->copy) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->copy(POINTER_OFFSET(src_data, src_offset), dest_data, 1);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(dest_data, POINTER_OFFSET(src_data, src_offset), typeInfo->size);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
/* if there are multiple source & dest layers of the same type,
|
|
|
|
* we don't want to copy all source layers to the same dest, so
|
|
|
|
* increment dest_i
|
|
|
|
*/
|
|
|
|
dest_i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (use_default_init) {
|
|
|
|
while (dest_i < dest->totlayer) {
|
|
|
|
CustomData_bmesh_set_default_n(dest, dest_block, dest_i);
|
|
|
|
dest_i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CustomData_from_bmesh_block(const CustomData *source,
|
|
|
|
CustomData *dest,
|
|
|
|
void *src_block,
|
|
|
|
int dst_index)
|
|
|
|
{
|
|
|
|
int dest_i, src_i;
|
|
|
|
|
|
|
|
/* copies a layer at a time */
|
|
|
|
dest_i = 0;
|
|
|
|
for (src_i = 0; src_i < source->totlayer; ++src_i) {
|
|
|
|
|
|
|
|
/* find the first dest layer with type >= the source type
|
|
|
|
* (this should work because layers are ordered by type)
|
|
|
|
*/
|
|
|
|
while (dest_i < dest->totlayer && dest->layers[dest_i].type < source->layers[src_i].type) {
|
|
|
|
dest_i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if there are no more dest layers, we're done */
|
2019-04-22 09:39:35 +10:00
|
|
|
if (dest_i >= dest->totlayer) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
/* if we found a matching layer, copy the data */
|
|
|
|
if (dest->layers[dest_i].type == source->layers[src_i].type) {
|
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(dest->layers[dest_i].type);
|
|
|
|
int offset = source->layers[src_i].offset;
|
|
|
|
const void *src_data = POINTER_OFFSET(src_block, offset);
|
|
|
|
void *dst_data = POINTER_OFFSET(dest->layers[dest_i].data,
|
|
|
|
(size_t)dst_index * typeInfo->size);
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->copy) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->copy(src_data, dst_data, 1);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
memcpy(dst_data, src_data, typeInfo->size);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
/* if there are multiple source & dest layers of the same type,
|
|
|
|
* we don't want to copy all source layers to the same dest, so
|
|
|
|
* increment dest_i
|
|
|
|
*/
|
|
|
|
dest_i++;
|
|
|
|
}
|
|
|
|
}
|
2006-11-11 16:38:37 +00:00
|
|
|
}
|
|
|
|
|
2016-05-17 03:07:40 +10:00
|
|
|
void CustomData_file_write_info(int type, const char **r_struct_name, int *r_struct_num)
|
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
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
*r_struct_name = typeInfo->structname;
|
|
|
|
*r_struct_num = typeInfo->structnum;
|
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
|
|
|
}
|
|
|
|
|
2015-07-21 12:02:11 +02:00
|
|
|
/**
|
|
|
|
* Prepare given custom data for file writing.
|
|
|
|
*
|
2018-12-12 12:50:58 +11:00
|
|
|
* \param data: the customdata to tweak for .blend file writing (modified in place).
|
2019-04-27 12:07:07 +10:00
|
|
|
* \param r_write_layers: contains a reduced set of layers to be written to file,
|
|
|
|
* use it with writestruct_at_address()
|
|
|
|
* (caller must free it if != \a write_layers_buff).
|
|
|
|
*
|
2018-12-12 12:50:58 +11:00
|
|
|
* \param write_layers_buff: an optional buffer for r_write_layers (to avoid allocating it).
|
|
|
|
* \param write_layers_size: the size of pre-allocated \a write_layer_buff.
|
2015-07-21 12:02:11 +02:00
|
|
|
*
|
2019-04-27 12:07:07 +10:00
|
|
|
* \warning After this func has ran, given custom data is no more valid from Blender PoV
|
|
|
|
* (its totlayer is invalid). This func shall always be called with localized data
|
|
|
|
* (as it is in write_meshes()).
|
|
|
|
*
|
|
|
|
* \note data->typemap is not updated here, since it is always rebuilt on file read anyway.
|
|
|
|
* This means written typemap does not match written layers (as returned by \a r_write_layers).
|
|
|
|
* Trivial to fix is ever needed.
|
2015-07-21 12:02:11 +02:00
|
|
|
*/
|
2019-04-17 06:17:24 +02:00
|
|
|
void CustomData_file_write_prepare(CustomData *data,
|
|
|
|
CustomDataLayer **r_write_layers,
|
|
|
|
CustomDataLayer *write_layers_buff,
|
|
|
|
size_t write_layers_size)
|
|
|
|
{
|
|
|
|
CustomDataLayer *write_layers = write_layers_buff;
|
|
|
|
const size_t chunk_size = (write_layers_size > 0) ? write_layers_size : CD_TEMP_CHUNK_SIZE;
|
|
|
|
|
|
|
|
const int totlayer = data->totlayer;
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
for (i = 0, j = 0; i < totlayer; i++) {
|
|
|
|
CustomDataLayer *layer = &data->layers[i];
|
|
|
|
if (layer->flag & CD_FLAG_NOCOPY) { /* Layers with this flag set are not written to file. */
|
|
|
|
data->totlayer--;
|
|
|
|
/* CLOG_WARN(&LOG, "skipping layer %p (%s)", layer, layer->name); */
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (UNLIKELY((size_t)j >= write_layers_size)) {
|
|
|
|
if (write_layers == write_layers_buff) {
|
|
|
|
write_layers = MEM_malloc_arrayN(
|
|
|
|
(write_layers_size + chunk_size), sizeof(*write_layers), __func__);
|
|
|
|
if (write_layers_buff) {
|
|
|
|
memcpy(write_layers, write_layers_buff, sizeof(*write_layers) * write_layers_size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
write_layers = MEM_reallocN(write_layers,
|
|
|
|
sizeof(*write_layers) * (write_layers_size + chunk_size));
|
|
|
|
}
|
|
|
|
write_layers_size += chunk_size;
|
|
|
|
}
|
|
|
|
write_layers[j++] = *layer;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
BLI_assert(j == data->totlayer);
|
|
|
|
data->maxlayer = data->totlayer; /* We only write that much of data! */
|
|
|
|
*r_write_layers = write_layers;
|
2015-07-21 12:02:11 +02: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
|
|
|
int CustomData_sizeof(int type)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
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
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return typeInfo->size;
|
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
|
|
|
}
|
2006-12-05 17:42:03 +00:00
|
|
|
|
|
|
|
const char *CustomData_layertype_name(int type)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
return layerType_getName(type);
|
2006-12-05 17:42:03 +00:00
|
|
|
}
|
2006-12-21 13:47:27 +00:00
|
|
|
|
2012-04-30 18:54:14 +00:00
|
|
|
/**
|
|
|
|
* Can only ever be one of these.
|
|
|
|
*/
|
2013-05-08 12:57:18 +00:00
|
|
|
bool CustomData_layertype_is_singleton(int type)
|
2012-04-30 18:54:14 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
|
|
|
return typeInfo->defaultname == NULL;
|
2012-04-30 18:54:14 +00:00
|
|
|
}
|
|
|
|
|
2014-10-21 17:01:56 +02:00
|
|
|
/**
|
|
|
|
* \return Maximum number of layers of given \a type, -1 means 'no limit'.
|
|
|
|
*/
|
|
|
|
int CustomData_layertype_layers_max(const int type)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(type);
|
2014-10-21 17:01:56 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* Same test as for singleton above. */
|
|
|
|
if (typeInfo->defaultname == NULL) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else if (typeInfo->layers_max == NULL) {
|
|
|
|
return -1;
|
|
|
|
}
|
2014-10-21 17:01:56 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return typeInfo->layers_max();
|
2014-10-21 17:01:56 +02:00
|
|
|
}
|
|
|
|
|
2013-05-08 12:57:18 +00:00
|
|
|
static bool CustomData_is_property_layer(int type)
|
2007-06-04 19:18:19 +00:00
|
|
|
{
|
2019-04-22 09:39:35 +10:00
|
|
|
if ((type == CD_PROP_FLT) || (type == CD_PROP_INT) || (type == CD_PROP_STR)) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return true;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2007-06-04 19:18:19 +00:00
|
|
|
}
|
|
|
|
|
2013-03-04 19:27:51 +00:00
|
|
|
static bool cd_layer_find_dupe(CustomData *data, const char *name, int type, int index)
|
2006-12-21 13:47:27 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int i;
|
|
|
|
/* see if there is a duplicate */
|
|
|
|
for (i = 0; i < data->totlayer; i++) {
|
|
|
|
if (i != index) {
|
|
|
|
CustomDataLayer *layer = &data->layers[i];
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (CustomData_is_property_layer(type)) {
|
|
|
|
if (CustomData_is_property_layer(layer->type) && STREQ(layer->name, name)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (i != index && layer->type == type && STREQ(layer->name, name)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-06-17 17:05:51 +02:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2010-11-01 07:19:41 +00:00
|
|
|
}
|
2006-12-21 13:47:27 +00:00
|
|
|
|
2013-03-04 19:27:51 +00:00
|
|
|
static bool customdata_unique_check(void *arg, const char *name)
|
2010-11-01 07:19:41 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
struct {
|
|
|
|
CustomData *data;
|
|
|
|
int type;
|
|
|
|
int index;
|
|
|
|
} *data_arg = arg;
|
|
|
|
return cd_layer_find_dupe(data_arg->data, name, data_arg->type, data_arg->index);
|
2010-11-07 08:49:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CustomData_set_layer_unique_name(CustomData *data, int index)
|
2018-06-17 17:05:51 +02:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomDataLayer *nlayer = &data->layers[index];
|
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(nlayer->type);
|
2006-12-21 13:47:27 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
struct {
|
|
|
|
CustomData *data;
|
|
|
|
int type;
|
|
|
|
int index;
|
|
|
|
} data_arg;
|
|
|
|
data_arg.data = data;
|
|
|
|
data_arg.type = nlayer->type;
|
|
|
|
data_arg.index = index;
|
2010-11-07 08:49:07 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (!typeInfo->defaultname) {
|
|
|
|
return;
|
|
|
|
}
|
2018-11-24 19:21:35 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* Set default name if none specified. Note we only call DATA_() when
|
|
|
|
* needed to avoid overhead of locale lookups in the depsgraph. */
|
|
|
|
if (nlayer->name[0] == '\0') {
|
|
|
|
STRNCPY(nlayer->name, DATA_(typeInfo->defaultname));
|
|
|
|
}
|
2013-03-25 08:29:06 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
BLI_uniquename_cb(
|
|
|
|
customdata_unique_check, &data_arg, NULL, '.', nlayer->name, sizeof(nlayer->name));
|
2006-12-21 13:47:27 +00:00
|
|
|
}
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
void CustomData_validate_layer_name(const CustomData *data,
|
|
|
|
int type,
|
|
|
|
const char *name,
|
|
|
|
char *outname)
|
2011-10-13 20:00:22 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
int index = -1;
|
2011-10-13 20:00:22 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
/* if a layer name was given, try to find that layer */
|
2019-04-22 09:39:35 +10:00
|
|
|
if (name[0]) {
|
2019-04-17 06:17:24 +02:00
|
|
|
index = CustomData_get_named_layer_index(data, type, name);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2011-10-13 20:00:22 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (index == -1) {
|
|
|
|
/* either no layer was specified, or the layer we want has been
|
|
|
|
* deleted, so assign the active layer to name
|
|
|
|
*/
|
|
|
|
index = CustomData_get_active_layer_index(data, type);
|
|
|
|
BLI_strncpy(outname, data->layers[index].name, MAX_CUSTOMDATA_LAYER_NAME);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
BLI_strncpy(outname, name, MAX_CUSTOMDATA_LAYER_NAME);
|
|
|
|
}
|
2011-10-13 20:00:22 +00:00
|
|
|
}
|
|
|
|
|
2013-05-08 12:57:18 +00:00
|
|
|
bool CustomData_verify_versions(struct CustomData *data, int index)
|
2007-01-06 20:16:06 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo;
|
|
|
|
CustomDataLayer *layer = &data->layers[index];
|
|
|
|
bool keeplayer = true;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (layer->type >= CD_NUMTYPES) {
|
|
|
|
keeplayer = false; /* unknown layer type from future version */
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
typeInfo = layerType_getInfo(layer->type);
|
|
|
|
|
|
|
|
if (!typeInfo->defaultname && (index > 0) && data->layers[index - 1].type == layer->type) {
|
|
|
|
keeplayer = false; /* multiple layers of which we only support one */
|
|
|
|
}
|
2019-04-27 12:07:07 +10:00
|
|
|
/* This is a pre-emptive fix for cases that should not happen
|
|
|
|
* (layers that should not be written in .blend files),
|
|
|
|
* but can happen due to bugs (see e.g. T62318).
|
|
|
|
* Also for forward compatibility, in future,
|
|
|
|
* we may put into `.blend` file some currently un-written data types,
|
2019-04-17 06:17:24 +02:00
|
|
|
* this should cover that case as well.
|
|
|
|
* Better to be safe here, and fix issue on the fly rather than crash... */
|
|
|
|
/* 0 structnum is used in writing code to tag layer types that should not be written. */
|
|
|
|
else if (typeInfo->structnum == 0 &&
|
|
|
|
/* XXX Not sure why those two are exception, maybe that should be fixed? */
|
|
|
|
!ELEM(layer->type, CD_PAINT_MASK, CD_FACEMAP)) {
|
|
|
|
keeplayer = false;
|
|
|
|
CLOG_WARN(&LOG, ".blend file read: removing a data layer that should not have been written");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!keeplayer) {
|
2019-04-22 09:39:35 +10:00
|
|
|
for (i = index + 1; i < data->totlayer; ++i) {
|
2019-04-17 06:17:24 +02:00
|
|
|
data->layers[i - 1] = data->layers[i];
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
data->totlayer--;
|
|
|
|
}
|
|
|
|
|
|
|
|
return keeplayer;
|
2007-01-06 20:16:06 +00:00
|
|
|
}
|
|
|
|
|
2018-12-03 16:19:08 +01:00
|
|
|
/**
|
2019-04-27 12:07:07 +10:00
|
|
|
* Validate and fix data of \a layer,
|
|
|
|
* if possible (needs relevant callback in layer's type to be defined).
|
2018-12-03 16:19:08 +01:00
|
|
|
*
|
|
|
|
* \return True if some errors were found.
|
|
|
|
*/
|
|
|
|
bool CustomData_layer_validate(CustomDataLayer *layer, const uint totitems, const bool do_fixes)
|
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
const LayerTypeInfo *typeInfo = layerType_getInfo(layer->type);
|
2018-12-03 16:19:08 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (typeInfo->validate != NULL) {
|
|
|
|
return typeInfo->validate(layer->data, totitems, do_fixes);
|
|
|
|
}
|
2018-12-03 16:19:08 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2018-12-03 16:19:08 +01:00
|
|
|
}
|
|
|
|
|
2009-11-25 14:27:50 +00:00
|
|
|
/****************************** External Files *******************************/
|
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
static void customdata_external_filename(char filename[FILE_MAX],
|
|
|
|
ID *id,
|
|
|
|
CustomDataExternal *external)
|
2009-11-25 14:27:50 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
BLI_strncpy(filename, external->filename, FILE_MAX);
|
|
|
|
BLI_path_abs(filename, ID_BLEND_PATH_FROM_GLOBAL(id));
|
2009-11-25 14:27:50 +00:00
|
|
|
}
|
|
|
|
|
2010-10-17 06:38:56 +00:00
|
|
|
void CustomData_external_reload(CustomData *data, ID *UNUSED(id), CustomDataMask mask, int totelem)
|
2010-06-01 19:26:35 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomDataLayer *layer;
|
|
|
|
const LayerTypeInfo *typeInfo;
|
|
|
|
int i;
|
2010-06-01 19:26:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
for (i = 0; i < data->totlayer; i++) {
|
|
|
|
layer = &data->layers[i];
|
|
|
|
typeInfo = layerType_getInfo(layer->type);
|
2010-06-01 19:26:35 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (!(mask & CD_TYPE_AS_MASK(layer->type))) {
|
|
|
|
/* pass */
|
|
|
|
}
|
|
|
|
else if ((layer->flag & CD_FLAG_EXTERNAL) && (layer->flag & CD_FLAG_IN_MEMORY)) {
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->free) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->free(layer->data, totelem, typeInfo->size);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
layer->flag &= ~CD_FLAG_IN_MEMORY;
|
|
|
|
}
|
|
|
|
}
|
2009-11-25 14:27:50 +00:00
|
|
|
}
|
|
|
|
|
2009-12-10 14:26:06 +00:00
|
|
|
void CustomData_external_read(CustomData *data, ID *id, CustomDataMask mask, int totelem)
|
2009-11-25 14:27:50 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomDataExternal *external = data->external;
|
|
|
|
CustomDataLayer *layer;
|
|
|
|
CDataFile *cdf;
|
|
|
|
CDataFileLayer *blay;
|
|
|
|
char filename[FILE_MAX];
|
|
|
|
const LayerTypeInfo *typeInfo;
|
|
|
|
int i, update = 0;
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!external) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
for (i = 0; i < data->totlayer; i++) {
|
|
|
|
layer = &data->layers[i];
|
|
|
|
typeInfo = layerType_getInfo(layer->type);
|
|
|
|
|
|
|
|
if (!(mask & CD_TYPE_AS_MASK(layer->type))) {
|
|
|
|
/* pass */
|
|
|
|
}
|
|
|
|
else if (layer->flag & CD_FLAG_IN_MEMORY) {
|
|
|
|
/* pass */
|
|
|
|
}
|
|
|
|
else if ((layer->flag & CD_FLAG_EXTERNAL) && typeInfo->read) {
|
|
|
|
update = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!update) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
customdata_external_filename(filename, id, external);
|
|
|
|
|
|
|
|
cdf = cdf_create(CDF_TYPE_MESH);
|
|
|
|
if (!cdf_read_open(cdf, filename)) {
|
|
|
|
cdf_free(cdf);
|
|
|
|
CLOG_ERROR(&LOG, "Failed to read %s layer from %s.", layerType_getName(layer->type), filename);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < data->totlayer; i++) {
|
|
|
|
layer = &data->layers[i];
|
|
|
|
typeInfo = layerType_getInfo(layer->type);
|
|
|
|
|
|
|
|
if (!(mask & CD_TYPE_AS_MASK(layer->type))) {
|
|
|
|
/* pass */
|
|
|
|
}
|
|
|
|
else if (layer->flag & CD_FLAG_IN_MEMORY) {
|
|
|
|
/* pass */
|
|
|
|
}
|
|
|
|
else if ((layer->flag & CD_FLAG_EXTERNAL) && typeInfo->read) {
|
|
|
|
blay = cdf_layer_find(cdf, layer->type, layer->name);
|
|
|
|
|
|
|
|
if (blay) {
|
|
|
|
if (cdf_read_layer(cdf, blay)) {
|
|
|
|
if (typeInfo->read(cdf, layer->data, totelem)) {
|
|
|
|
/* pass */
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
layer->flag |= CD_FLAG_IN_MEMORY;
|
|
|
|
}
|
2019-04-22 09:39:35 +10:00
|
|
|
else {
|
2019-04-17 06:17:24 +02:00
|
|
|
break;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cdf_read_close(cdf);
|
|
|
|
cdf_free(cdf);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CustomData_external_write(
|
|
|
|
CustomData *data, ID *id, CustomDataMask mask, int totelem, int free)
|
|
|
|
{
|
|
|
|
CustomDataExternal *external = data->external;
|
|
|
|
CustomDataLayer *layer;
|
|
|
|
CDataFile *cdf;
|
|
|
|
CDataFileLayer *blay;
|
|
|
|
const LayerTypeInfo *typeInfo;
|
|
|
|
int i, update = 0;
|
|
|
|
char filename[FILE_MAX];
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!external) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
/* test if there is anything to write */
|
|
|
|
for (i = 0; i < data->totlayer; i++) {
|
|
|
|
layer = &data->layers[i];
|
|
|
|
typeInfo = layerType_getInfo(layer->type);
|
|
|
|
|
|
|
|
if (!(mask & CD_TYPE_AS_MASK(layer->type))) {
|
|
|
|
/* pass */
|
|
|
|
}
|
|
|
|
else if ((layer->flag & CD_FLAG_EXTERNAL) && typeInfo->write) {
|
|
|
|
update = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!update) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
/* make sure data is read before we try to write */
|
|
|
|
CustomData_external_read(data, id, mask, totelem);
|
|
|
|
customdata_external_filename(filename, id, external);
|
|
|
|
|
|
|
|
cdf = cdf_create(CDF_TYPE_MESH);
|
|
|
|
|
|
|
|
for (i = 0; i < data->totlayer; i++) {
|
|
|
|
layer = &data->layers[i];
|
|
|
|
typeInfo = layerType_getInfo(layer->type);
|
|
|
|
|
|
|
|
if ((layer->flag & CD_FLAG_EXTERNAL) && typeInfo->filesize) {
|
|
|
|
if (layer->flag & CD_FLAG_IN_MEMORY) {
|
|
|
|
cdf_layer_add(
|
|
|
|
cdf, layer->type, layer->name, typeInfo->filesize(cdf, layer->data, totelem));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
cdf_free(cdf);
|
|
|
|
return; /* read failed for a layer! */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!cdf_write_open(cdf, filename)) {
|
|
|
|
CLOG_ERROR(&LOG, "Failed to open %s for writing.", filename);
|
|
|
|
cdf_free(cdf);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < data->totlayer; i++) {
|
|
|
|
layer = &data->layers[i];
|
|
|
|
typeInfo = layerType_getInfo(layer->type);
|
|
|
|
|
|
|
|
if ((layer->flag & CD_FLAG_EXTERNAL) && typeInfo->write) {
|
|
|
|
blay = cdf_layer_find(cdf, layer->type, layer->name);
|
|
|
|
|
|
|
|
if (cdf_write_layer(cdf, blay)) {
|
|
|
|
if (typeInfo->write(cdf, layer->data, totelem)) {
|
|
|
|
/* pass */
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i != data->totlayer) {
|
|
|
|
CLOG_ERROR(&LOG, "Failed to write data to %s.", filename);
|
|
|
|
cdf_write_close(cdf);
|
|
|
|
cdf_free(cdf);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < data->totlayer; i++) {
|
|
|
|
layer = &data->layers[i];
|
|
|
|
typeInfo = layerType_getInfo(layer->type);
|
|
|
|
|
|
|
|
if ((layer->flag & CD_FLAG_EXTERNAL) && typeInfo->write) {
|
|
|
|
if (free) {
|
2019-04-22 09:39:35 +10:00
|
|
|
if (typeInfo->free) {
|
2019-04-17 06:17:24 +02:00
|
|
|
typeInfo->free(layer->data, totelem, typeInfo->size);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
layer->flag &= ~CD_FLAG_IN_MEMORY;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cdf_write_close(cdf);
|
|
|
|
cdf_free(cdf);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CustomData_external_add(
|
|
|
|
CustomData *data, ID *UNUSED(id), int type, int UNUSED(totelem), const char *filename)
|
|
|
|
{
|
|
|
|
CustomDataExternal *external = data->external;
|
|
|
|
CustomDataLayer *layer;
|
|
|
|
int layer_index;
|
|
|
|
|
|
|
|
layer_index = CustomData_get_active_layer_index(data, type);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer_index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
layer = &data->layers[layer_index];
|
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer->flag & CD_FLAG_EXTERNAL) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2019-04-17 06:17:24 +02:00
|
|
|
|
|
|
|
if (!external) {
|
|
|
|
external = MEM_callocN(sizeof(CustomDataExternal), "CustomDataExternal");
|
|
|
|
data->external = external;
|
|
|
|
}
|
|
|
|
BLI_strncpy(external->filename, filename, sizeof(external->filename));
|
|
|
|
|
|
|
|
layer->flag |= CD_FLAG_EXTERNAL | CD_FLAG_IN_MEMORY;
|
2009-11-25 14:27:50 +00:00
|
|
|
}
|
|
|
|
|
2009-12-10 14:26:06 +00:00
|
|
|
void CustomData_external_remove(CustomData *data, ID *id, int type, int totelem)
|
2009-11-25 14:27:50 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomDataExternal *external = data->external;
|
|
|
|
CustomDataLayer *layer;
|
2019-05-01 11:09:22 +10:00
|
|
|
// char filename[FILE_MAX];
|
2019-04-17 06:17:24 +02:00
|
|
|
int layer_index; // i, remove_file;
|
2009-11-25 14:27:50 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
layer_index = CustomData_get_active_layer_index(data, type);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer_index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2009-11-25 14:27:50 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
layer = &data->layers[layer_index];
|
2009-11-25 14:27:50 +00:00
|
|
|
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!external) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2009-11-25 14:27:50 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
if (layer->flag & CD_FLAG_EXTERNAL) {
|
2019-04-22 09:39:35 +10:00
|
|
|
if (!(layer->flag & CD_FLAG_IN_MEMORY)) {
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomData_external_read(data, id, CD_TYPE_AS_MASK(layer->type), totelem);
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2009-11-25 14:27:50 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
layer->flag &= ~CD_FLAG_EXTERNAL;
|
|
|
|
}
|
2009-11-25 14:27:50 +00:00
|
|
|
}
|
|
|
|
|
2013-05-08 12:57:18 +00:00
|
|
|
bool CustomData_external_test(CustomData *data, int type)
|
2009-11-25 14:27:50 +00:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
CustomDataLayer *layer;
|
|
|
|
int layer_index;
|
2009-11-25 14:27:50 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
layer_index = CustomData_get_active_layer_index(data, type);
|
2019-04-22 09:39:35 +10:00
|
|
|
if (layer_index == -1) {
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
2019-04-22 09:39:35 +10:00
|
|
|
}
|
2009-11-25 14:27:50 +00:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
layer = &data->layers[layer_index];
|
|
|
|
return (layer->flag & CD_FLAG_EXTERNAL) != 0;
|
2009-11-25 14:27:50 +00:00
|
|
|
}
|
|
|
|
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
/* ********** Mesh-to-mesh data transfer ********** */
|
2015-02-23 13:51:55 +11:00
|
|
|
static void copy_bit_flag(void *dst, const void *src, const size_t data_size, const uint64_t flag)
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
#define COPY_BIT_FLAG(_type, _dst, _src, _f) \
|
|
|
|
{ \
|
|
|
|
const _type _val = *((_type *)(_src)) & ((_type)(_f)); \
|
|
|
|
*((_type *)(_dst)) &= ~((_type)(_f)); \
|
|
|
|
*((_type *)(_dst)) |= _val; \
|
|
|
|
} \
|
|
|
|
(void)0
|
|
|
|
|
|
|
|
switch (data_size) {
|
|
|
|
case 1:
|
|
|
|
COPY_BIT_FLAG(uint8_t, dst, src, flag);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
COPY_BIT_FLAG(uint16_t, dst, src, flag);
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
COPY_BIT_FLAG(uint32_t, dst, src, flag);
|
|
|
|
break;
|
|
|
|
case 8:
|
|
|
|
COPY_BIT_FLAG(uint64_t, dst, src, flag);
|
|
|
|
break;
|
|
|
|
default:
|
2019-05-01 11:09:22 +10:00
|
|
|
// CLOG_ERROR(&LOG, "Unknown flags-container size (%zu)", datasize);
|
2019-04-17 06:17:24 +02:00
|
|
|
break;
|
|
|
|
}
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
|
|
|
|
#undef COPY_BIT_FLAG
|
|
|
|
}
|
|
|
|
|
2015-02-23 13:51:55 +11:00
|
|
|
static bool check_bit_flag(const void *data, const size_t data_size, const uint64_t flag)
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
{
|
2019-04-17 06:17:24 +02:00
|
|
|
switch (data_size) {
|
|
|
|
case 1:
|
|
|
|
return ((*((uint8_t *)data) & ((uint8_t)flag)) != 0);
|
|
|
|
case 2:
|
|
|
|
return ((*((uint16_t *)data) & ((uint16_t)flag)) != 0);
|
|
|
|
case 4:
|
|
|
|
return ((*((uint32_t *)data) & ((uint32_t)flag)) != 0);
|
|
|
|
case 8:
|
|
|
|
return ((*((uint64_t *)data) & ((uint64_t)flag)) != 0);
|
|
|
|
default:
|
2019-05-01 11:09:22 +10:00
|
|
|
// CLOG_ERROR(&LOG, "Unknown flags-container size (%zu)", datasize);
|
2019-04-17 06:17:24 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void customdata_data_transfer_interp_generic(const CustomDataTransferLayerMap *laymap,
|
|
|
|
void *data_dst,
|
|
|
|
const void **sources,
|
|
|
|
const float *weights,
|
|
|
|
const int count,
|
|
|
|
const float mix_factor)
|
|
|
|
{
|
|
|
|
/* Fake interpolation, we actually copy highest weighted source to dest.
|
2019-04-27 12:07:07 +10:00
|
|
|
* Note we also handle bitflags here,
|
|
|
|
* in which case we rather choose to transfer value of elements totaling
|
2019-04-17 06:17:24 +02:00
|
|
|
* more than 0.5 of weight. */
|
|
|
|
int best_src_idx = 0;
|
|
|
|
|
|
|
|
const int data_type = laymap->data_type;
|
|
|
|
const int mix_mode = laymap->mix_mode;
|
|
|
|
|
|
|
|
size_t data_size;
|
|
|
|
const uint64_t data_flag = laymap->data_flag;
|
|
|
|
|
|
|
|
cd_interp interp_cd = NULL;
|
|
|
|
cd_copy copy_cd = NULL;
|
|
|
|
|
|
|
|
void *tmp_dst;
|
|
|
|
|
|
|
|
if (!sources) {
|
|
|
|
/* Not supported here, abort. */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data_type & CD_FAKE) {
|
|
|
|
data_size = laymap->data_size;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
const LayerTypeInfo *type_info = layerType_getInfo(data_type);
|
|
|
|
|
|
|
|
data_size = (size_t)type_info->size;
|
|
|
|
interp_cd = type_info->interp;
|
|
|
|
copy_cd = type_info->copy;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp_dst = MEM_mallocN(data_size, __func__);
|
|
|
|
|
|
|
|
if (count > 1 && !interp_cd) {
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (data_flag) {
|
2019-04-27 12:07:07 +10:00
|
|
|
/* Boolean case, we can 'interpolate' in two groups,
|
|
|
|
* and choose value from highest weighted group. */
|
2019-04-17 06:17:24 +02:00
|
|
|
float tot_weight_true = 0.0f;
|
|
|
|
int item_true_idx = -1, item_false_idx = -1;
|
|
|
|
|
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
if (check_bit_flag(sources[i], data_size, data_flag)) {
|
|
|
|
tot_weight_true += weights[i];
|
|
|
|
item_true_idx = i;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
item_false_idx = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
best_src_idx = (tot_weight_true >= 0.5f) ? item_true_idx : item_false_idx;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* We just choose highest weighted source. */
|
|
|
|
float max_weight = 0.0f;
|
|
|
|
|
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
if (weights[i] > max_weight) {
|
|
|
|
max_weight = weights[i];
|
|
|
|
best_src_idx = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
BLI_assert(best_src_idx >= 0);
|
|
|
|
|
|
|
|
if (interp_cd) {
|
|
|
|
interp_cd(sources, weights, NULL, count, tmp_dst);
|
|
|
|
}
|
|
|
|
else if (data_flag) {
|
|
|
|
copy_bit_flag(tmp_dst, sources[best_src_idx], data_size, data_flag);
|
|
|
|
}
|
|
|
|
/* No interpolation, just copy highest weight source element's data. */
|
|
|
|
else if (copy_cd) {
|
|
|
|
copy_cd(sources[best_src_idx], tmp_dst, 1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
memcpy(tmp_dst, sources[best_src_idx], data_size);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data_flag) {
|
2019-04-27 12:07:07 +10:00
|
|
|
/* Bool flags, only copy if dest data is set (resp. unset) -
|
|
|
|
* only 'advanced' modes we can support here! */
|
2019-04-17 06:17:24 +02:00
|
|
|
if (mix_factor >= 0.5f && ((mix_mode == CDT_MIX_TRANSFER) ||
|
|
|
|
(mix_mode == CDT_MIX_REPLACE_ABOVE_THRESHOLD &&
|
|
|
|
check_bit_flag(data_dst, data_size, data_flag)) ||
|
|
|
|
(mix_mode == CDT_MIX_REPLACE_BELOW_THRESHOLD &&
|
|
|
|
!check_bit_flag(data_dst, data_size, data_flag)))) {
|
|
|
|
copy_bit_flag(data_dst, tmp_dst, data_size, data_flag);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (!(data_type & CD_FAKE)) {
|
|
|
|
CustomData_data_mix_value(data_type, tmp_dst, data_dst, mix_mode, mix_factor);
|
|
|
|
}
|
|
|
|
/* Else we can do nothing by default, needs custom interp func!
|
|
|
|
* Note this is here only for sake of consistency, not expected to be used much actually? */
|
|
|
|
else {
|
|
|
|
if (mix_factor >= 0.5f) {
|
|
|
|
memcpy(data_dst, tmp_dst, data_size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
MEM_freeN(tmp_dst);
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
}
|
|
|
|
|
2015-10-16 21:28:22 +02:00
|
|
|
/* Normals are special, we need to take care of source & destination spaces... */
|
2019-04-17 06:17:24 +02:00
|
|
|
void customdata_data_transfer_interp_normal_normals(const CustomDataTransferLayerMap *laymap,
|
|
|
|
void *data_dst,
|
|
|
|
const void **sources,
|
|
|
|
const float *weights,
|
|
|
|
const int count,
|
|
|
|
const float mix_factor)
|
|
|
|
{
|
|
|
|
const int data_type = laymap->data_type;
|
|
|
|
const int mix_mode = laymap->mix_mode;
|
|
|
|
|
|
|
|
SpaceTransform *space_transform = laymap->interp_data;
|
|
|
|
|
|
|
|
const LayerTypeInfo *type_info = layerType_getInfo(data_type);
|
|
|
|
cd_interp interp_cd = type_info->interp;
|
|
|
|
|
|
|
|
float tmp_dst[3];
|
|
|
|
|
|
|
|
BLI_assert(data_type == CD_NORMAL);
|
|
|
|
|
|
|
|
if (!sources) {
|
|
|
|
/* Not supported here, abort. */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
interp_cd(sources, weights, NULL, count, tmp_dst);
|
|
|
|
if (space_transform) {
|
|
|
|
/* tmp_dst is in source space so far, bring it back in destination space. */
|
|
|
|
BLI_space_transform_invert_normal(space_transform, tmp_dst);
|
|
|
|
}
|
|
|
|
|
|
|
|
CustomData_data_mix_value(data_type, tmp_dst, data_dst, mix_mode, mix_factor);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CustomData_data_transfer(const MeshPairRemap *me_remap,
|
|
|
|
const CustomDataTransferLayerMap *laymap)
|
|
|
|
{
|
|
|
|
MeshPairRemapItem *mapit = me_remap->items;
|
|
|
|
const int totelem = me_remap->items_num;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
const int data_type = laymap->data_type;
|
|
|
|
const void *data_src = laymap->data_src;
|
|
|
|
void *data_dst = laymap->data_dst;
|
|
|
|
|
|
|
|
size_t data_step;
|
|
|
|
size_t data_size;
|
|
|
|
size_t data_offset;
|
|
|
|
|
|
|
|
cd_datatransfer_interp interp = NULL;
|
|
|
|
|
|
|
|
size_t tmp_buff_size = 32;
|
|
|
|
const void **tmp_data_src = NULL;
|
|
|
|
|
|
|
|
/* Note: NULL data_src may happen and be valid (see vgroups...). */
|
|
|
|
if (!data_dst) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data_src) {
|
|
|
|
tmp_data_src = MEM_malloc_arrayN(tmp_buff_size, sizeof(*tmp_data_src), __func__);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data_type & CD_FAKE) {
|
|
|
|
data_step = laymap->elem_size;
|
|
|
|
data_size = laymap->data_size;
|
|
|
|
data_offset = laymap->data_offset;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
const LayerTypeInfo *type_info = layerType_getInfo(data_type);
|
|
|
|
|
|
|
|
/* Note: we can use 'fake' CDLayers, like e.g. for crease, bweight, etc. :/ */
|
|
|
|
data_size = (size_t)type_info->size;
|
|
|
|
data_step = laymap->elem_size ? laymap->elem_size : data_size;
|
|
|
|
data_offset = laymap->data_offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
interp = laymap->interp ? laymap->interp : customdata_data_transfer_interp_generic;
|
|
|
|
|
|
|
|
for (i = 0; i < totelem; i++, data_dst = POINTER_OFFSET(data_dst, data_step), mapit++) {
|
|
|
|
const int sources_num = mapit->sources_num;
|
|
|
|
const float mix_factor = laymap->mix_weights ? laymap->mix_weights[i] : laymap->mix_factor;
|
|
|
|
int j;
|
|
|
|
|
|
|
|
if (!sources_num) {
|
|
|
|
/* No sources for this element, skip it. */
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tmp_data_src) {
|
|
|
|
if (UNLIKELY(sources_num > tmp_buff_size)) {
|
|
|
|
tmp_buff_size = (size_t)sources_num;
|
|
|
|
tmp_data_src = MEM_reallocN((void *)tmp_data_src, sizeof(*tmp_data_src) * tmp_buff_size);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (j = 0; j < sources_num; j++) {
|
|
|
|
const size_t src_idx = (size_t)mapit->indices_src[j];
|
|
|
|
tmp_data_src[j] = POINTER_OFFSET(data_src, (data_step * src_idx) + data_offset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
interp(laymap,
|
|
|
|
POINTER_OFFSET(data_dst, data_offset),
|
|
|
|
tmp_data_src,
|
|
|
|
mapit->weights_src,
|
|
|
|
sources_num,
|
|
|
|
mix_factor);
|
|
|
|
}
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
|
2019-04-17 06:17:24 +02:00
|
|
|
MEM_SAFE_FREE(tmp_data_src);
|
Transfer Data: add main core code and operators.
This add code needed to map a CD data layout from source mesh towards destination one,
and code needed to actually transfer data, using BKE's mesh remap generated data.
This allows to transfer most CD layers (vgroups, vcols, uvs...) as well as fake, boolean ones
(like smooth/sharp edges/faces, etc.). Some types are not yet transferable, mainly
shape keys, this is known TODO.
Data transfer can also use some advanced mixing in some cases (mostly, vgroups and vcols).
Notes:
* New transfer operators transfer data from active object towards selected ones.
* Modifier will be committed separately.
* Old weight transfer code (for vgroups) is kept for now, mostly because it is the only
usable one in weightpaint mode (it transfers from selected object to active one,
this is not sensible in Object mode, but needed in WeightPaint one). This will be addressed soon.
Again, heavily reviewed and enhanced by Campbell, thanks!
2015-01-09 19:11:40 +01:00
|
|
|
}
|