2011-10-10 09:38:02 +00:00
|
|
|
/*
|
2002-10-12 11:37:38 +00:00
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
2008-01-07 19:13:47 +00:00
|
|
|
* of the License, or (at your option) any later version.
|
2002-10-12 11:37:38 +00:00
|
|
|
*
|
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
|
* along with this program; if not, write to the Free Software Foundation,
|
2010-02-12 13:34:04 +00:00
|
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2002-10-12 11:37:38 +00:00
|
|
|
*
|
|
|
|
|
* The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
|
|
|
|
|
* All rights reserved.
|
|
|
|
|
*/
|
2002-11-25 10:13:52 +00:00
|
|
|
|
2019-02-18 08:08:12 +11:00
|
|
|
/** \file
|
|
|
|
|
* \ingroup bke
|
2011-02-27 20:40:57 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
2002-10-12 11:37:38 +00:00
|
|
|
#include <math.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
|
|
|
|
|
|
#include "DNA_curve_types.h"
|
2018-06-05 15:59:30 +02:00
|
|
|
#include "DNA_mesh_types.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "DNA_scene_types.h"
|
2010-08-04 04:01:27 +00:00
|
|
|
#include "DNA_object_types.h"
|
2014-01-03 14:18:06 +11:00
|
|
|
#include "DNA_vfont_types.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
|
#include "BLI_blenlib.h"
|
2013-08-28 02:07:54 +00:00
|
|
|
#include "BLI_memarena.h"
|
2009-11-10 20:43:45 +00:00
|
|
|
#include "BLI_math.h"
|
2010-11-03 01:56:02 +00:00
|
|
|
#include "BLI_scanfill.h"
|
2011-01-07 18:36:47 +00:00
|
|
|
#include "BLI_utildefines.h"
|
2019-02-14 17:21:55 +11:00
|
|
|
#include "BLI_linklist.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
|
|
|
|
|
#include "BKE_displist.h"
|
2010-03-05 16:47:52 +00:00
|
|
|
#include "BKE_cdderivedmesh.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "BKE_object.h"
|
2018-06-05 15:59:30 +02:00
|
|
|
#include "BKE_library.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "BKE_mball.h"
|
2015-03-25 22:36:43 +11:00
|
|
|
#include "BKE_mball_tessellate.h"
|
2018-06-05 15:59:30 +02:00
|
|
|
#include "BKE_mesh.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "BKE_curve.h"
|
2005-08-14 06:08:41 +00:00
|
|
|
#include "BKE_key.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
#include "BKE_anim.h"
|
|
|
|
|
#include "BKE_font.h"
|
|
|
|
|
#include "BKE_lattice.h"
|
2005-08-14 12:17:34 +00:00
|
|
|
#include "BKE_modifier.h"
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2013-05-28 19:35:26 +00:00
|
|
|
#include "BLI_sys_types.h" // for intptr_t support
|
2008-08-17 17:08:00 +00:00
|
|
|
|
2017-04-06 15:37:46 +02:00
|
|
|
#include "DEG_depsgraph.h"
|
2017-04-06 16:11:50 +02:00
|
|
|
#include "DEG_depsgraph_query.h"
|
2017-04-06 15:37:46 +02:00
|
|
|
|
2013-08-19 09:13:15 +00:00
|
|
|
static void boundbox_displist_object(Object *ob);
|
2005-07-17 04:17:33 +00:00
|
|
|
|
2012-05-07 06:58:03 +00:00
|
|
|
void BKE_displist_elem_free(DispList *dl)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
2012-03-24 06:18:31 +00:00
|
|
|
if (dl) {
|
|
|
|
|
if (dl->verts) MEM_freeN(dl->verts);
|
|
|
|
|
if (dl->nors) MEM_freeN(dl->nors);
|
|
|
|
|
if (dl->index) MEM_freeN(dl->index);
|
2016-06-25 11:24:25 +10:00
|
|
|
if (dl->bevel_split) MEM_freeN(dl->bevel_split);
|
2002-10-12 11:37:38 +00:00
|
|
|
MEM_freeN(dl);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-05-07 06:58:03 +00:00
|
|
|
void BKE_displist_free(ListBase *lb)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
|
DispList *dl;
|
|
|
|
|
|
2013-08-26 23:37:08 +00:00
|
|
|
while ((dl = BLI_pophead(lb))) {
|
2012-05-07 06:58:03 +00:00
|
|
|
BKE_displist_elem_free(dl);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-05-07 06:58:03 +00:00
|
|
|
DispList *BKE_displist_find_or_create(ListBase *lb, int type)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
|
DispList *dl;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = lb->first;
|
2012-03-24 06:18:31 +00:00
|
|
|
while (dl) {
|
2012-05-07 06:58:03 +00:00
|
|
|
if (dl->type == type)
|
|
|
|
|
return dl;
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = dl->next;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = MEM_callocN(sizeof(DispList), "find_disp");
|
|
|
|
|
dl->type = type;
|
2002-10-12 11:37:38 +00:00
|
|
|
BLI_addtail(lb, dl);
|
|
|
|
|
|
|
|
|
|
return dl;
|
|
|
|
|
}
|
|
|
|
|
|
2012-05-07 06:58:03 +00:00
|
|
|
DispList *BKE_displist_find(ListBase *lb, int type)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
|
DispList *dl;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = lb->first;
|
2012-03-24 06:18:31 +00:00
|
|
|
while (dl) {
|
2012-05-07 06:58:03 +00:00
|
|
|
if (dl->type == type)
|
|
|
|
|
return dl;
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = dl->next;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
|
2011-02-13 10:52:18 +00:00
|
|
|
return NULL;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
|
2013-03-20 23:14:18 +00:00
|
|
|
bool BKE_displist_has_faces(ListBase *lb)
|
2004-10-29 15:39:01 +00:00
|
|
|
{
|
|
|
|
|
DispList *dl;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
for (dl = lb->first; dl; dl = dl->next) {
|
2014-07-20 01:30:29 +10:00
|
|
|
if (ELEM(dl->type, DL_INDEX3, DL_INDEX4, DL_SURF)) {
|
2013-03-20 23:14:18 +00:00
|
|
|
return true;
|
2012-03-25 22:35:18 +00:00
|
|
|
}
|
2004-10-29 15:39:01 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2013-03-20 23:14:18 +00:00
|
|
|
return false;
|
2004-10-29 15:39:01 +00:00
|
|
|
}
|
|
|
|
|
|
2012-05-07 06:58:03 +00:00
|
|
|
void BKE_displist_copy(ListBase *lbn, ListBase *lb)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
|
DispList *dln, *dl;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
|
|
|
|
BKE_displist_free(lbn);
|
|
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = lb->first;
|
2012-03-24 06:18:31 +00:00
|
|
|
while (dl) {
|
2012-05-06 17:22:54 +00:00
|
|
|
dln = MEM_dupallocN(dl);
|
2002-10-12 11:37:38 +00:00
|
|
|
BLI_addtail(lbn, dln);
|
2012-05-06 17:22:54 +00:00
|
|
|
dln->verts = MEM_dupallocN(dl->verts);
|
|
|
|
|
dln->nors = MEM_dupallocN(dl->nors);
|
|
|
|
|
dln->index = MEM_dupallocN(dl->index);
|
2011-01-03 19:45:08 +00:00
|
|
|
|
2016-06-25 11:24:25 +10:00
|
|
|
if (dl->bevel_split) {
|
|
|
|
|
dln->bevel_split = MEM_dupallocN(dl->bevel_split);
|
|
|
|
|
}
|
2011-01-03 19:45:08 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = dl->next;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-05-07 06:58:03 +00:00
|
|
|
void BKE_displist_normals_add(ListBase *lb)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
|
DispList *dl = NULL;
|
|
|
|
|
float *vdata, *ndata, nor[3];
|
|
|
|
|
float *v1, *v2, *v3, *v4;
|
|
|
|
|
float *n1, *n2, *n3, *n4;
|
|
|
|
|
int a, b, p1, p2, p3, p4;
|
|
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = lb->first;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
while (dl) {
|
2012-05-06 17:22:54 +00:00
|
|
|
if (dl->type == DL_INDEX3) {
|
|
|
|
|
if (dl->nors == NULL) {
|
|
|
|
|
dl->nors = MEM_callocN(sizeof(float) * 3, "dlnors");
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2017-05-26 19:03:30 +10:00
|
|
|
if (dl->flag & DL_BACK_CURVE) {
|
2012-05-07 06:58:03 +00:00
|
|
|
dl->nors[2] = -1.0f;
|
2017-05-26 19:03:30 +10:00
|
|
|
}
|
|
|
|
|
else {
|
2012-05-07 06:58:03 +00:00
|
|
|
dl->nors[2] = 1.0f;
|
2017-05-26 19:03:30 +10:00
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|
2012-05-06 17:22:54 +00:00
|
|
|
else if (dl->type == DL_SURF) {
|
|
|
|
|
if (dl->nors == NULL) {
|
|
|
|
|
dl->nors = MEM_callocN(sizeof(float) * 3 * dl->nr * dl->parts, "dlnors");
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
vdata = dl->verts;
|
|
|
|
|
ndata = dl->nors;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
for (a = 0; a < dl->parts; a++) {
|
2012-05-07 06:58:03 +00:00
|
|
|
|
|
|
|
|
if (BKE_displist_surfindex_get(dl, a, &b, &p1, &p2, &p3, &p4) == 0)
|
2008-10-06 06:10:14 +00:00
|
|
|
break;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
v1 = vdata + 3 * p1;
|
|
|
|
|
n1 = ndata + 3 * p1;
|
|
|
|
|
v2 = vdata + 3 * p2;
|
|
|
|
|
n2 = ndata + 3 * p2;
|
|
|
|
|
v3 = vdata + 3 * p3;
|
|
|
|
|
n3 = ndata + 3 * p3;
|
|
|
|
|
v4 = vdata + 3 * p4;
|
|
|
|
|
n4 = ndata + 3 * p4;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
for (; b < dl->nr; b++) {
|
2012-04-29 15:47:02 +00:00
|
|
|
normal_quad_v3(nor, v1, v3, v4, v2);
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2010-04-21 12:27:48 +00:00
|
|
|
add_v3_v3(n1, nor);
|
|
|
|
|
add_v3_v3(n2, nor);
|
|
|
|
|
add_v3_v3(n3, nor);
|
|
|
|
|
add_v3_v3(n4, nor);
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
v2 = v1; v1 += 3;
|
|
|
|
|
v4 = v3; v3 += 3;
|
|
|
|
|
n2 = n1; n1 += 3;
|
|
|
|
|
n4 = n3; n3 += 3;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|
2012-05-06 17:22:54 +00:00
|
|
|
a = dl->parts * dl->nr;
|
|
|
|
|
v1 = ndata;
|
2012-03-24 06:18:31 +00:00
|
|
|
while (a--) {
|
2009-11-10 20:43:45 +00:00
|
|
|
normalize_v3(v1);
|
2012-05-06 17:22:54 +00:00
|
|
|
v1 += 3;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = dl->next;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-12-27 03:51:45 +00:00
|
|
|
void BKE_displist_count(ListBase *lb, int *totvert, int *totface, int *tottri)
|
2006-05-28 11:49:22 +00:00
|
|
|
{
|
|
|
|
|
DispList *dl;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-12-27 03:51:45 +00:00
|
|
|
for (dl = lb->first; dl; dl = dl->next) {
|
|
|
|
|
int vert_tot = 0;
|
|
|
|
|
int face_tot = 0;
|
|
|
|
|
int tri_tot = 0;
|
|
|
|
|
|
2012-04-28 06:31:57 +00:00
|
|
|
switch (dl->type) {
|
2006-05-28 11:49:22 +00:00
|
|
|
case DL_SURF:
|
2012-12-27 03:51:45 +00:00
|
|
|
{
|
|
|
|
|
vert_tot = dl->nr * dl->parts;
|
|
|
|
|
face_tot = (dl->nr - 1) * (dl->parts - 1);
|
|
|
|
|
tri_tot = face_tot * 2;
|
2006-05-28 11:49:22 +00:00
|
|
|
break;
|
2012-12-27 03:51:45 +00:00
|
|
|
}
|
2006-05-28 11:49:22 +00:00
|
|
|
case DL_INDEX3:
|
2012-12-27 03:51:45 +00:00
|
|
|
{
|
|
|
|
|
vert_tot = dl->nr;
|
|
|
|
|
face_tot = dl->parts;
|
|
|
|
|
tri_tot = face_tot;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2006-05-28 11:49:22 +00:00
|
|
|
case DL_INDEX4:
|
2012-12-27 03:51:45 +00:00
|
|
|
{
|
|
|
|
|
vert_tot = dl->nr;
|
|
|
|
|
face_tot = dl->parts;
|
|
|
|
|
tri_tot = face_tot * 2;
|
2006-05-28 11:49:22 +00:00
|
|
|
break;
|
2012-12-27 03:51:45 +00:00
|
|
|
}
|
2006-05-28 11:49:22 +00:00
|
|
|
case DL_POLY:
|
|
|
|
|
case DL_SEGM:
|
2012-12-27 03:51:45 +00:00
|
|
|
{
|
|
|
|
|
vert_tot = dl->nr * dl->parts;
|
2012-05-07 06:58:03 +00:00
|
|
|
break;
|
2012-12-27 03:51:45 +00:00
|
|
|
}
|
2006-05-28 11:49:22 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-12-27 03:51:45 +00:00
|
|
|
*totvert += vert_tot;
|
|
|
|
|
*totface += face_tot;
|
|
|
|
|
*tottri += tri_tot;
|
2006-05-28 11:49:22 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-03-20 23:14:18 +00:00
|
|
|
bool BKE_displist_surfindex_get(DispList *dl, int a, int *b, int *p1, int *p2, int *p3, int *p4)
|
2008-10-06 06:10:14 +00:00
|
|
|
{
|
2012-05-06 17:22:54 +00:00
|
|
|
if ((dl->flag & DL_CYCL_V) == 0 && a == (dl->parts) - 1) {
|
2013-03-20 23:14:18 +00:00
|
|
|
return false;
|
2008-10-06 06:10:14 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (dl->flag & DL_CYCL_U) {
|
2012-05-06 17:22:54 +00:00
|
|
|
(*p1) = dl->nr * a;
|
|
|
|
|
(*p2) = (*p1) + dl->nr - 1;
|
|
|
|
|
(*p3) = (*p1) + dl->nr;
|
|
|
|
|
(*p4) = (*p2) + dl->nr;
|
|
|
|
|
(*b) = 0;
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
|
|
|
|
else {
|
2012-05-06 17:22:54 +00:00
|
|
|
(*p2) = dl->nr * a;
|
|
|
|
|
(*p1) = (*p2) + 1;
|
|
|
|
|
(*p4) = (*p2) + dl->nr;
|
|
|
|
|
(*p3) = (*p1) + dl->nr;
|
|
|
|
|
(*b) = 1;
|
2008-10-06 06:10:14 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
|
|
|
|
if ((dl->flag & DL_CYCL_V) && a == dl->parts - 1) {
|
2012-05-06 17:22:54 +00:00
|
|
|
(*p3) -= dl->nr * dl->parts;
|
|
|
|
|
(*p4) -= dl->nr * dl->parts;
|
2008-10-06 06:10:14 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2013-03-20 23:14:18 +00:00
|
|
|
return true;
|
2008-10-06 06:10:14 +00:00
|
|
|
}
|
2006-05-28 11:49:22 +00:00
|
|
|
|
|
|
|
|
/* ****************** make displists ********************* */
|
2018-04-02 16:11:11 +02:00
|
|
|
#ifdef __INTEL_COMPILER
|
|
|
|
|
/* ICC with the optimization -02 causes crashes. */
|
|
|
|
|
# pragma intel optimization_level 1
|
|
|
|
|
#endif
|
2014-03-20 22:56:28 +11:00
|
|
|
static void curve_to_displist(Curve *cu, ListBase *nubase, ListBase *dispbase,
|
|
|
|
|
const bool for_render, const bool use_render_resolution)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
|
Nurb *nu;
|
|
|
|
|
DispList *dl;
|
|
|
|
|
BezTriple *bezt, *prevbezt;
|
|
|
|
|
BPoint *bp;
|
2009-09-10 02:57:25 +00:00
|
|
|
float *data;
|
2006-02-07 19:59:02 +00:00
|
|
|
int a, len, resolu;
|
2014-03-20 22:56:28 +11:00
|
|
|
const bool editmode = (!for_render && (cu->editnurb || cu->editfont));
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
nu = nubase->first;
|
2012-03-24 06:18:31 +00:00
|
|
|
while (nu) {
|
2014-03-20 22:56:28 +11:00
|
|
|
if (nu->hide == 0 || editmode == false) {
|
|
|
|
|
if (use_render_resolution && cu->resolu_ren != 0)
|
2012-05-06 17:22:54 +00:00
|
|
|
resolu = cu->resolu_ren;
|
2006-02-07 19:59:02 +00:00
|
|
|
else
|
2012-05-06 17:22:54 +00:00
|
|
|
resolu = nu->resolu;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-10-07 09:48:59 +00:00
|
|
|
if (!BKE_nurb_check_valid_u(nu)) {
|
|
|
|
|
/* pass */
|
|
|
|
|
}
|
2012-03-24 06:18:31 +00:00
|
|
|
else if (nu->type == CU_BEZIER) {
|
2003-04-25 15:48:11 +00:00
|
|
|
/* count */
|
2012-05-06 17:22:54 +00:00
|
|
|
len = 0;
|
|
|
|
|
a = nu->pntsu - 1;
|
2012-03-24 06:18:31 +00:00
|
|
|
if (nu->flagu & CU_NURB_CYCLIC) a++;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
prevbezt = nu->bezt;
|
|
|
|
|
bezt = prevbezt + 1;
|
2012-03-24 06:18:31 +00:00
|
|
|
while (a--) {
|
2012-05-07 06:58:03 +00:00
|
|
|
if (a == 0 && (nu->flagu & CU_NURB_CYCLIC))
|
|
|
|
|
bezt = nu->bezt;
|
|
|
|
|
|
|
|
|
|
if (prevbezt->h2 == HD_VECT && bezt->h1 == HD_VECT)
|
|
|
|
|
len++;
|
|
|
|
|
else
|
|
|
|
|
len += resolu;
|
|
|
|
|
|
|
|
|
|
if (a == 0 && (nu->flagu & CU_NURB_CYCLIC) == 0)
|
|
|
|
|
len++;
|
|
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
prevbezt = bezt;
|
2002-10-12 11:37:38 +00:00
|
|
|
bezt++;
|
|
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = MEM_callocN(sizeof(DispList), "makeDispListbez");
|
2005-07-16 19:07:02 +00:00
|
|
|
/* len+1 because of 'forward_diff_bezier' function */
|
2014-06-15 14:36:33 +10:00
|
|
|
dl->verts = MEM_mallocN((len + 1) * sizeof(float[3]), "dlverts");
|
2002-10-12 11:37:38 +00:00
|
|
|
BLI_addtail(dispbase, dl);
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->parts = 1;
|
|
|
|
|
dl->nr = len;
|
|
|
|
|
dl->col = nu->mat_nr;
|
|
|
|
|
dl->charidx = nu->charidx;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
data = dl->verts;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2015-02-27 17:43:10 +11:00
|
|
|
/* check that (len != 2) so we don't immediately loop back on ourselves */
|
|
|
|
|
if (nu->flagu & CU_NURB_CYCLIC && (dl->nr != 2)) {
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->type = DL_POLY;
|
|
|
|
|
a = nu->pntsu;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
else {
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->type = DL_SEGM;
|
|
|
|
|
a = nu->pntsu - 1;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
prevbezt = nu->bezt;
|
|
|
|
|
bezt = prevbezt + 1;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
while (a--) {
|
2012-05-07 06:58:03 +00:00
|
|
|
if (a == 0 && dl->type == DL_POLY)
|
|
|
|
|
bezt = nu->bezt;
|
|
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
if (prevbezt->h2 == HD_VECT && bezt->h1 == HD_VECT) {
|
2011-11-06 15:17:43 +00:00
|
|
|
copy_v3_v3(data, prevbezt->vec[1]);
|
2012-05-06 17:22:54 +00:00
|
|
|
data += 3;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
else {
|
2009-09-10 02:57:25 +00:00
|
|
|
int j;
|
2012-05-06 17:22:54 +00:00
|
|
|
for (j = 0; j < 3; j++) {
|
2012-04-29 17:11:40 +00:00
|
|
|
BKE_curve_forward_diff_bezier(prevbezt->vec[1][j],
|
|
|
|
|
prevbezt->vec[2][j],
|
|
|
|
|
bezt->vec[0][j],
|
|
|
|
|
bezt->vec[1][j],
|
|
|
|
|
data + j, resolu, 3 * sizeof(float));
|
2009-09-10 02:57:25 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
data += 3 * resolu;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
if (a == 0 && dl->type == DL_SEGM) {
|
2011-11-06 15:17:43 +00:00
|
|
|
copy_v3_v3(data, bezt->vec[1]);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
prevbezt = bezt;
|
2002-10-12 11:37:38 +00:00
|
|
|
bezt++;
|
|
|
|
|
}
|
|
|
|
|
}
|
2012-03-24 06:18:31 +00:00
|
|
|
else if (nu->type == CU_NURBS) {
|
2012-05-06 17:22:54 +00:00
|
|
|
len = (resolu * SEGMENTSU(nu));
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = MEM_callocN(sizeof(DispList), "makeDispListsurf");
|
2014-06-19 15:13:15 +10:00
|
|
|
dl->verts = MEM_mallocN(len * sizeof(float[3]), "dlverts");
|
2002-10-12 11:37:38 +00:00
|
|
|
BLI_addtail(dispbase, dl);
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->parts = 1;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->nr = len;
|
|
|
|
|
dl->col = nu->mat_nr;
|
2006-02-02 12:13:08 +00:00
|
|
|
dl->charidx = nu->charidx;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
data = dl->verts;
|
2012-05-07 06:58:03 +00:00
|
|
|
if (nu->flagu & CU_NURB_CYCLIC)
|
|
|
|
|
dl->type = DL_POLY;
|
2012-05-06 17:22:54 +00:00
|
|
|
else dl->type = DL_SEGM;
|
|
|
|
|
BKE_nurb_makeCurve(nu, data, NULL, NULL, NULL, resolu, 3 * sizeof(float));
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-03-24 06:18:31 +00:00
|
|
|
else if (nu->type == CU_POLY) {
|
2012-05-06 17:22:54 +00:00
|
|
|
len = nu->pntsu;
|
|
|
|
|
dl = MEM_callocN(sizeof(DispList), "makeDispListpoly");
|
2014-06-15 14:36:33 +10:00
|
|
|
dl->verts = MEM_mallocN(len * sizeof(float[3]), "dlverts");
|
2002-10-12 11:37:38 +00:00
|
|
|
BLI_addtail(dispbase, dl);
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->parts = 1;
|
|
|
|
|
dl->nr = len;
|
|
|
|
|
dl->col = nu->mat_nr;
|
More text object fancyness, and fixes:
- "Flush" is now split into two seperate Alignment modes "Flush" and
"Justify":
- Justify does exactly the same as a normal word processor's justify
function does, and in addition, it uses *whitespace* instead of
*character spacing* (kerning) to fill lines. Much more readable.
- Flush is pretty much the old Blender "Flush" mode - and as such it
uses character spacing to fill lines. Just as Justify, this only
works with at least one textframe.
- Underlining for text objects. Not a lot to explain. New button "U" in
the editbuttons, and CTRL-U as hotkey toggle underlining for newly
entered characters or for the selection, just like CTRL-B/CTRL-I do for
bold/italic.
Underline height (thickness) and Underline position (vertical) can be
set in the editbuttons.
Implemented as CU_POLY polygon curves.
- The B, U and i buttons (and the corresponding CTRL-B/U/I keystrokes)
have been fixed to only affect *one* attribute at a time. Formerly,
hitting CTRL-B when no other style was active, on a text portion with
italics text, for example, would kill the italics and just apply bold.
Now, these attributes always add or substract only, but do not
replace the style.
- In the past, there were bugs with material indices uninitialized, and
thus crashes in the renderer with illegal material indices.
Even though I assume they have been fixed, I've put in a check that
checks (hah) if the material index of a character is illegal (bigger
than ob->totcol), and then sets it to zero, and spits out a warning
on stderr.
If you see such warnings, please report and link to the .blend.
- Bugfix: All alignment modes only worked if there were at least *two*
lines of text in the text object. Fixed
There's now a regression test file for text objects, please add to the
corresponding repository:
http://blender.instinctive.de/downloads/release/demo/text-regression.blend.gz
2005-08-29 12:46:07 +00:00
|
|
|
dl->charidx = nu->charidx;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
data = dl->verts;
|
2015-02-27 17:43:10 +11:00
|
|
|
if ((nu->flagu & CU_NURB_CYCLIC) && (dl->nr != 2)) {
|
|
|
|
|
dl->type = DL_POLY;
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
dl->type = DL_SEGM;
|
|
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
a = len;
|
|
|
|
|
bp = nu->bp;
|
2012-03-24 06:18:31 +00:00
|
|
|
while (a--) {
|
2011-11-06 15:17:43 +00:00
|
|
|
copy_v3_v3(data, bp->vec);
|
2002-10-12 11:37:38 +00:00
|
|
|
bp++;
|
2012-05-06 17:22:54 +00:00
|
|
|
data += 3;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2012-05-06 17:22:54 +00:00
|
|
|
nu = nu->next;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-04-26 21:04:12 +00:00
|
|
|
/**
|
2018-12-12 12:50:58 +11:00
|
|
|
* \param normal_proj: Optional normal thats used to project the scanfill verts into 2d coords.
|
2013-04-26 21:04:12 +00:00
|
|
|
* Pass this along if known since it saves time calculating the normal.
|
2018-12-12 12:50:58 +11:00
|
|
|
* \param flipnormal: Flip the normal (same as passing \a normal_proj negated)
|
2013-04-26 21:04:12 +00:00
|
|
|
*/
|
|
|
|
|
void BKE_displist_fill(ListBase *dispbase, ListBase *to, const float normal_proj[3], const bool flipnormal)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
2012-04-16 06:48:57 +00:00
|
|
|
ScanFillContext sf_ctx;
|
2012-05-13 14:47:53 +00:00
|
|
|
ScanFillVert *sf_vert, *sf_vert_new, *sf_vert_last;
|
|
|
|
|
ScanFillFace *sf_tri;
|
2013-08-28 02:07:54 +00:00
|
|
|
MemArena *sf_arena;
|
2012-05-06 17:22:54 +00:00
|
|
|
DispList *dlnew = NULL, *dl;
|
2002-10-12 11:37:38 +00:00
|
|
|
float *f1;
|
2012-05-06 17:22:54 +00:00
|
|
|
int colnr = 0, charidx = 0, cont = 1, tot, a, *index, nextcol = 0;
|
2014-02-06 01:07:18 +11:00
|
|
|
int totvert;
|
2014-02-13 19:09:28 +11:00
|
|
|
const int scanfill_flag = BLI_SCANFILL_CALC_REMOVE_DOUBLES | BLI_SCANFILL_CALC_POLYS | BLI_SCANFILL_CALC_HOLES;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
|
|
|
|
if (dispbase == NULL)
|
|
|
|
|
return;
|
2014-02-08 06:07:10 +11:00
|
|
|
if (BLI_listbase_is_empty(dispbase))
|
2012-05-07 06:58:03 +00:00
|
|
|
return;
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2013-08-28 02:07:54 +00:00
|
|
|
sf_arena = BLI_memarena_new(BLI_SCANFILL_ARENA_SIZE, __func__);
|
|
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
while (cont) {
|
2017-05-26 19:03:30 +10:00
|
|
|
int dl_flag_accum = 0;
|
2012-05-06 17:22:54 +00:00
|
|
|
cont = 0;
|
|
|
|
|
totvert = 0;
|
|
|
|
|
nextcol = 0;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2013-08-28 02:07:54 +00:00
|
|
|
BLI_scanfill_begin_arena(&sf_ctx, sf_arena);
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = dispbase->first;
|
2012-03-24 06:18:31 +00:00
|
|
|
while (dl) {
|
2012-05-06 17:22:54 +00:00
|
|
|
if (dl->type == DL_POLY) {
|
2012-05-07 06:58:03 +00:00
|
|
|
if (charidx < dl->charidx)
|
|
|
|
|
cont = 1;
|
2012-05-06 17:22:54 +00:00
|
|
|
else if (charidx == dl->charidx) { /* character with needed index */
|
|
|
|
|
if (colnr == dl->col) {
|
2014-02-05 05:22:21 +11:00
|
|
|
|
|
|
|
|
sf_ctx.poly_nr++;
|
|
|
|
|
|
2010-09-22 05:08:52 +00:00
|
|
|
/* make editverts and edges */
|
2012-05-06 17:22:54 +00:00
|
|
|
f1 = dl->verts;
|
|
|
|
|
a = dl->nr;
|
2012-05-13 14:47:53 +00:00
|
|
|
sf_vert = sf_vert_new = NULL;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
while (a--) {
|
2012-05-13 14:47:53 +00:00
|
|
|
sf_vert_last = sf_vert;
|
2010-09-22 05:08:52 +00:00
|
|
|
|
2012-05-13 14:47:53 +00:00
|
|
|
sf_vert = BLI_scanfill_vert_add(&sf_ctx, f1);
|
2010-09-22 05:08:52 +00:00
|
|
|
totvert++;
|
|
|
|
|
|
2012-05-13 14:47:53 +00:00
|
|
|
if (sf_vert_last == NULL)
|
|
|
|
|
sf_vert_new = sf_vert;
|
2010-09-22 05:08:52 +00:00
|
|
|
else {
|
2012-05-13 14:47:53 +00:00
|
|
|
BLI_scanfill_edge_add(&sf_ctx, sf_vert_last, sf_vert);
|
2010-09-22 05:08:52 +00:00
|
|
|
}
|
2012-05-06 17:22:54 +00:00
|
|
|
f1 += 3;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2010-09-22 05:08:52 +00:00
|
|
|
|
2012-05-13 14:47:53 +00:00
|
|
|
if (sf_vert != NULL && sf_vert_new != NULL) {
|
|
|
|
|
BLI_scanfill_edge_add(&sf_ctx, sf_vert, sf_vert_new);
|
2010-09-22 05:08:52 +00:00
|
|
|
}
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
2012-05-06 17:22:54 +00:00
|
|
|
else if (colnr < dl->col) {
|
2010-09-22 05:08:52 +00:00
|
|
|
/* got poly with next material at current char */
|
2012-05-06 17:22:54 +00:00
|
|
|
cont = 1;
|
|
|
|
|
nextcol = 1;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|
2017-05-26 19:03:30 +10:00
|
|
|
dl_flag_accum |= dl->flag;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = dl->next;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2015-06-17 07:06:59 +10:00
|
|
|
/* XXX (obedit && obedit->actcol) ? (obedit->actcol - 1) : 0)) { */
|
2013-04-26 21:04:12 +00:00
|
|
|
if (totvert && (tot = BLI_scanfill_calc_ex(&sf_ctx,
|
2014-02-13 19:09:28 +11:00
|
|
|
scanfill_flag,
|
2013-04-26 21:04:12 +00:00
|
|
|
normal_proj)))
|
|
|
|
|
{
|
2012-03-24 06:18:31 +00:00
|
|
|
if (tot) {
|
2012-05-06 17:22:54 +00:00
|
|
|
dlnew = MEM_callocN(sizeof(DispList), "filldisplist");
|
|
|
|
|
dlnew->type = DL_INDEX3;
|
2017-05-26 19:03:30 +10:00
|
|
|
dlnew->flag = (dl_flag_accum & (DL_BACK_CURVE | DL_FRONT_CURVE));
|
2012-05-06 17:22:54 +00:00
|
|
|
dlnew->col = colnr;
|
|
|
|
|
dlnew->nr = totvert;
|
|
|
|
|
dlnew->parts = tot;
|
|
|
|
|
|
|
|
|
|
dlnew->index = MEM_mallocN(tot * 3 * sizeof(int), "dlindex");
|
|
|
|
|
dlnew->verts = MEM_mallocN(totvert * 3 * sizeof(float), "dlverts");
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* vert data */
|
2012-05-06 17:22:54 +00:00
|
|
|
f1 = dlnew->verts;
|
|
|
|
|
totvert = 0;
|
2012-07-11 18:17:48 +00:00
|
|
|
|
|
|
|
|
for (sf_vert = sf_ctx.fillvertbase.first; sf_vert; sf_vert = sf_vert->next) {
|
2012-05-13 14:47:53 +00:00
|
|
|
copy_v3_v3(f1, sf_vert->co);
|
2012-05-06 17:22:54 +00:00
|
|
|
f1 += 3;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* index number */
|
2014-02-06 01:07:18 +11:00
|
|
|
sf_vert->tmp.i = totvert;
|
2002-10-12 11:37:38 +00:00
|
|
|
totvert++;
|
|
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* index data */
|
2014-02-06 01:07:18 +11:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
index = dlnew->index;
|
2014-02-06 01:07:18 +11:00
|
|
|
for (sf_tri = sf_ctx.fillfacebase.first; sf_tri; sf_tri = sf_tri->next) {
|
|
|
|
|
index[0] = sf_tri->v1->tmp.i;
|
|
|
|
|
index[1] = sf_tri->v2->tmp.i;
|
|
|
|
|
index[2] = sf_tri->v3->tmp.i;
|
2010-06-23 13:18:50 +00:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (flipnormal)
|
2010-06-23 13:18:50 +00:00
|
|
|
SWAP(int, index[0], index[2]);
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
index += 3;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
BLI_addhead(to, dlnew);
|
|
|
|
|
}
|
2013-08-28 02:07:54 +00:00
|
|
|
BLI_scanfill_end_arena(&sf_ctx, sf_arena);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (nextcol) {
|
2010-09-22 05:08:52 +00:00
|
|
|
/* stay at current char but fill polys with next material */
|
|
|
|
|
colnr++;
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
|
|
|
|
else {
|
2010-09-22 05:08:52 +00:00
|
|
|
/* switch to next char and start filling from first material */
|
|
|
|
|
charidx++;
|
2012-05-06 17:22:54 +00:00
|
|
|
colnr = 0;
|
2010-09-22 05:08:52 +00:00
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2013-08-28 02:07:54 +00:00
|
|
|
BLI_memarena_free(sf_arena);
|
|
|
|
|
|
2003-04-25 15:48:11 +00:00
|
|
|
/* do not free polys, needed for wireframe display */
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void bevels_to_filledpoly(Curve *cu, ListBase *dispbase)
|
|
|
|
|
{
|
2013-04-26 21:04:12 +00:00
|
|
|
const float z_up[3] = {0.0f, 0.0f, 1.0f};
|
2002-10-12 11:37:38 +00:00
|
|
|
ListBase front, back;
|
|
|
|
|
DispList *dl, *dlnew;
|
|
|
|
|
float *fp, *fp1;
|
|
|
|
|
int a, dpoly;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2014-02-08 06:07:10 +11:00
|
|
|
BLI_listbase_clear(&front);
|
|
|
|
|
BLI_listbase_clear(&back);
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = dispbase->first;
|
2012-03-24 06:18:31 +00:00
|
|
|
while (dl) {
|
2012-05-06 17:22:54 +00:00
|
|
|
if (dl->type == DL_SURF) {
|
2012-05-07 06:58:03 +00:00
|
|
|
if ((dl->flag & DL_CYCL_V) && (dl->flag & DL_CYCL_U) == 0) {
|
2014-03-17 21:48:13 +11:00
|
|
|
if ((cu->flag & CU_BACK) && (dl->flag & DL_BACK_CURVE)) {
|
2012-05-06 17:22:54 +00:00
|
|
|
dlnew = MEM_callocN(sizeof(DispList), "filldisp");
|
2002-10-12 11:37:38 +00:00
|
|
|
BLI_addtail(&front, dlnew);
|
2012-05-06 17:22:54 +00:00
|
|
|
dlnew->verts = fp1 = MEM_mallocN(sizeof(float) * 3 * dl->parts, "filldisp1");
|
|
|
|
|
dlnew->nr = dl->parts;
|
|
|
|
|
dlnew->parts = 1;
|
|
|
|
|
dlnew->type = DL_POLY;
|
2017-05-26 19:03:30 +10:00
|
|
|
dlnew->flag = DL_BACK_CURVE;
|
2012-05-06 17:22:54 +00:00
|
|
|
dlnew->col = dl->col;
|
2006-02-02 12:13:08 +00:00
|
|
|
dlnew->charidx = dl->charidx;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
fp = dl->verts;
|
|
|
|
|
dpoly = 3 * dl->nr;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
a = dl->parts;
|
2012-03-24 06:18:31 +00:00
|
|
|
while (a--) {
|
2011-11-06 15:17:43 +00:00
|
|
|
copy_v3_v3(fp1, fp);
|
2012-05-06 17:22:54 +00:00
|
|
|
fp1 += 3;
|
|
|
|
|
fp += dpoly;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
if ((cu->flag & CU_FRONT) && (dl->flag & DL_FRONT_CURVE)) {
|
2012-05-06 17:22:54 +00:00
|
|
|
dlnew = MEM_callocN(sizeof(DispList), "filldisp");
|
2002-10-12 11:37:38 +00:00
|
|
|
BLI_addtail(&back, dlnew);
|
2012-05-06 17:22:54 +00:00
|
|
|
dlnew->verts = fp1 = MEM_mallocN(sizeof(float) * 3 * dl->parts, "filldisp1");
|
|
|
|
|
dlnew->nr = dl->parts;
|
|
|
|
|
dlnew->parts = 1;
|
|
|
|
|
dlnew->type = DL_POLY;
|
2017-05-26 19:03:30 +10:00
|
|
|
dlnew->flag = DL_FRONT_CURVE;
|
2012-05-06 17:22:54 +00:00
|
|
|
dlnew->col = dl->col;
|
|
|
|
|
dlnew->charidx = dl->charidx;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
fp = dl->verts + 3 * (dl->nr - 1);
|
|
|
|
|
dpoly = 3 * dl->nr;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
a = dl->parts;
|
2012-03-24 06:18:31 +00:00
|
|
|
while (a--) {
|
2011-11-06 15:17:43 +00:00
|
|
|
copy_v3_v3(fp1, fp);
|
2012-05-06 17:22:54 +00:00
|
|
|
fp1 += 3;
|
|
|
|
|
fp += dpoly;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = dl->next;
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
|
2013-04-26 21:04:12 +00:00
|
|
|
BKE_displist_fill(&front, dispbase, z_up, true);
|
|
|
|
|
BKE_displist_fill(&back, dispbase, z_up, false);
|
2012-05-07 06:58:03 +00:00
|
|
|
|
|
|
|
|
BKE_displist_free(&front);
|
|
|
|
|
BKE_displist_free(&back);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2013-04-26 21:04:12 +00:00
|
|
|
BKE_displist_fill(dispbase, dispbase, z_up, false);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
|
2010-10-16 14:32:17 +00:00
|
|
|
static void curve_to_filledpoly(Curve *cu, ListBase *UNUSED(nurb), ListBase *dispbase)
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
2013-12-23 12:12:40 +11:00
|
|
|
if (!CU_DO_2DFILL(cu))
|
2013-12-15 01:49:12 +11:00
|
|
|
return;
|
|
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
if (dispbase->first && ((DispList *) dispbase->first)->type == DL_SURF) {
|
2005-08-14 06:08:41 +00:00
|
|
|
bevels_to_filledpoly(cu, dispbase);
|
|
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
else {
|
2014-08-07 09:14:54 +10:00
|
|
|
const float z_up[3] = {0.0f, 0.0f, 1.0f};
|
|
|
|
|
BKE_displist_fill(dispbase, dispbase, z_up, false);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2004-08-29 09:21:52 +00:00
|
|
|
/* taper rules:
|
2012-03-09 18:28:30 +00:00
|
|
|
* - only 1 curve
|
|
|
|
|
* - first point left, last point right
|
|
|
|
|
* - based on subdivided points in original curve, not on points in taper curve (still)
|
|
|
|
|
*/
|
2018-04-06 12:07:27 +02:00
|
|
|
static float displist_calc_taper(Depsgraph *depsgraph, Scene *scene, Object *taperobj, float fac)
|
2004-08-29 09:21:52 +00:00
|
|
|
{
|
|
|
|
|
DispList *dl;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
|
|
|
|
if (taperobj == NULL || taperobj->type != OB_CURVE)
|
|
|
|
|
return 1.0;
|
|
|
|
|
|
2018-07-30 16:54:40 +02:00
|
|
|
dl = taperobj->runtime.curve_cache ? taperobj->runtime.curve_cache->disp.first : NULL;
|
2012-05-06 17:22:54 +00:00
|
|
|
if (dl == NULL) {
|
2019-02-14 17:21:55 +11:00
|
|
|
BKE_displist_make_curveTypes(depsgraph, scene, taperobj, false, false, NULL);
|
2018-07-30 16:54:40 +02:00
|
|
|
dl = taperobj->runtime.curve_cache->disp.first;
|
2004-08-29 09:21:52 +00:00
|
|
|
}
|
2012-03-24 06:18:31 +00:00
|
|
|
if (dl) {
|
2004-08-29 09:21:52 +00:00
|
|
|
float minx, dx, *fp;
|
|
|
|
|
int a;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2004-08-29 09:21:52 +00:00
|
|
|
/* horizontal size */
|
2012-05-06 17:22:54 +00:00
|
|
|
minx = dl->verts[0];
|
|
|
|
|
dx = dl->verts[3 * (dl->nr - 1)] - minx;
|
2012-03-24 06:18:31 +00:00
|
|
|
if (dx > 0.0f) {
|
2012-05-06 17:22:54 +00:00
|
|
|
fp = dl->verts;
|
|
|
|
|
for (a = 0; a < dl->nr; a++, fp += 3) {
|
2012-05-07 06:58:03 +00:00
|
|
|
if ((fp[0] - minx) / dx >= fac) {
|
2004-08-29 09:21:52 +00:00
|
|
|
/* interpolate with prev */
|
2012-05-06 17:22:54 +00:00
|
|
|
if (a > 0) {
|
|
|
|
|
float fac1 = (fp[-3] - minx) / dx;
|
|
|
|
|
float fac2 = (fp[0] - minx) / dx;
|
|
|
|
|
if (fac1 != fac2)
|
|
|
|
|
return fp[1] * (fac1 - fac) / (fac1 - fac2) + fp[-2] * (fac - fac2) / (fac1 - fac2);
|
2004-08-29 09:21:52 +00:00
|
|
|
}
|
|
|
|
|
return fp[1];
|
|
|
|
|
}
|
|
|
|
|
}
|
2012-05-06 17:22:54 +00:00
|
|
|
return fp[-2]; // last y coord
|
2004-08-29 09:21:52 +00:00
|
|
|
}
|
|
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2004-08-29 09:21:52 +00:00
|
|
|
return 1.0;
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-06 12:07:27 +02:00
|
|
|
float BKE_displist_calc_taper(Depsgraph *depsgraph, Scene *scene, Object *taperobj, int cur, int tot)
|
2012-11-08 08:16:44 +00:00
|
|
|
{
|
|
|
|
|
float fac = ((float)cur) / (float)(tot - 1);
|
|
|
|
|
|
2018-04-06 12:07:27 +02:00
|
|
|
return displist_calc_taper(depsgraph, scene, taperobj, fac);
|
2012-11-08 08:16:44 +00:00
|
|
|
}
|
|
|
|
|
|
2018-04-06 12:07:27 +02:00
|
|
|
void BKE_displist_make_mball(Depsgraph *depsgraph, Scene *scene, Object *ob)
|
2005-07-14 15:48:01 +00:00
|
|
|
{
|
2012-05-07 06:58:03 +00:00
|
|
|
if (!ob || ob->type != OB_MBALL)
|
|
|
|
|
return;
|
2005-07-14 15:48:01 +00:00
|
|
|
|
2014-07-18 17:46:45 +02:00
|
|
|
if (ob == BKE_mball_basis_find(scene, ob)) {
|
2018-07-30 16:54:40 +02:00
|
|
|
if (ob->runtime.curve_cache) {
|
|
|
|
|
BKE_displist_free(&(ob->runtime.curve_cache->disp));
|
2014-07-18 18:08:56 +02:00
|
|
|
}
|
|
|
|
|
else {
|
2018-07-30 16:54:40 +02:00
|
|
|
ob->runtime.curve_cache = MEM_callocN(sizeof(CurveCache), "CurveCache for MBall");
|
2014-07-18 18:08:56 +02:00
|
|
|
}
|
|
|
|
|
|
2018-07-30 16:54:40 +02:00
|
|
|
BKE_mball_polygonize(depsgraph, scene, ob, &ob->runtime.curve_cache->disp);
|
2014-07-18 17:46:45 +02:00
|
|
|
BKE_mball_texspace_calc(ob);
|
2003-10-14 12:09:57 +00:00
|
|
|
|
2018-07-30 16:54:40 +02:00
|
|
|
object_deform_mball(ob, &ob->runtime.curve_cache->disp);
|
2014-07-18 17:46:45 +02:00
|
|
|
|
2014-07-18 18:08:56 +02:00
|
|
|
/* NOP for MBALLs anyway... */
|
|
|
|
|
boundbox_displist_object(ob);
|
|
|
|
|
}
|
2005-07-14 15:48:01 +00:00
|
|
|
}
|
2005-08-14 12:17:34 +00:00
|
|
|
|
2018-04-06 12:07:27 +02:00
|
|
|
void BKE_displist_make_mball_forRender(Depsgraph *depsgraph, Scene *scene, Object *ob, ListBase *dispbase)
|
2010-06-28 11:07:02 +00:00
|
|
|
{
|
2018-04-06 12:07:27 +02:00
|
|
|
BKE_mball_polygonize(depsgraph, scene, ob, dispbase);
|
2012-05-07 06:38:41 +00:00
|
|
|
BKE_mball_texspace_calc(ob);
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2010-06-28 11:07:02 +00:00
|
|
|
object_deform_mball(ob, dispbase);
|
|
|
|
|
}
|
|
|
|
|
|
2014-03-20 22:56:28 +11:00
|
|
|
static ModifierData *curve_get_tessellate_point(Scene *scene, Object *ob,
|
|
|
|
|
const bool use_render_resolution, const bool editmode)
|
2005-08-14 12:17:34 +00:00
|
|
|
{
|
2013-08-19 09:05:34 +00:00
|
|
|
VirtualModifierData virtualModifierData;
|
|
|
|
|
ModifierData *md = modifiers_getVirtualModifierList(ob, &virtualModifierData);
|
2012-03-20 22:56:26 +00:00
|
|
|
ModifierData *pretessellatePoint;
|
2006-08-28 01:12:36 +00:00
|
|
|
int required_mode;
|
|
|
|
|
|
2014-03-20 22:56:28 +11:00
|
|
|
if (use_render_resolution)
|
2012-05-07 06:58:03 +00:00
|
|
|
required_mode = eModifierMode_Render;
|
|
|
|
|
else
|
|
|
|
|
required_mode = eModifierMode_Realtime;
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2012-05-07 06:58:03 +00:00
|
|
|
if (editmode)
|
|
|
|
|
required_mode |= eModifierMode_Editmode;
|
2005-08-14 12:17:34 +00:00
|
|
|
|
2012-03-20 22:56:26 +00:00
|
|
|
pretessellatePoint = NULL;
|
2012-05-06 17:22:54 +00:00
|
|
|
for (; md; md = md->next) {
|
2015-03-30 21:17:07 +11:00
|
|
|
const ModifierTypeInfo *mti = modifierType_getInfo(md->type);
|
2011-01-14 16:57:53 +00:00
|
|
|
|
2012-05-07 06:58:03 +00:00
|
|
|
if (!modifier_isEnabled(scene, md, required_mode))
|
|
|
|
|
continue;
|
|
|
|
|
if (mti->type == eModifierTypeType_Constructive)
|
|
|
|
|
return pretessellatePoint;
|
2005-08-14 12:17:34 +00:00
|
|
|
|
2014-07-20 01:30:29 +10:00
|
|
|
if (ELEM(md->type, eModifierType_Hook, eModifierType_Softbody, eModifierType_MeshDeform)) {
|
2012-03-20 22:56:26 +00:00
|
|
|
pretessellatePoint = md;
|
2011-01-14 16:57:53 +00:00
|
|
|
|
2012-03-02 16:05:54 +00:00
|
|
|
/* this modifiers are moving point of tessellation automatically
|
2012-03-18 07:38:51 +00:00
|
|
|
* (some of them even can't be applied on tessellated curve), set flag
|
2013-02-06 14:02:19 +00:00
|
|
|
* for information button in modifier's header
|
2012-05-07 06:58:03 +00:00
|
|
|
*/
|
2011-01-14 16:57:53 +00:00
|
|
|
md->mode |= eModifierMode_ApplyOnSpline;
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
2012-05-06 17:22:54 +00:00
|
|
|
else if (md->mode & eModifierMode_ApplyOnSpline) {
|
2012-03-20 22:56:26 +00:00
|
|
|
pretessellatePoint = md;
|
2005-08-14 12:17:34 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-03-20 22:56:26 +00:00
|
|
|
return pretessellatePoint;
|
2005-08-14 12:17:34 +00:00
|
|
|
}
|
|
|
|
|
|
2017-08-16 12:45:11 +10:00
|
|
|
static void curve_calc_modifiers_pre(
|
2018-04-06 12:07:27 +02:00
|
|
|
Depsgraph *depsgraph, Scene *scene, Object *ob, ListBase *nurb,
|
2017-08-16 12:45:11 +10:00
|
|
|
const bool for_render, const bool use_render_resolution)
|
2005-08-14 12:17:34 +00:00
|
|
|
{
|
2013-08-19 09:05:34 +00:00
|
|
|
VirtualModifierData virtualModifierData;
|
|
|
|
|
ModifierData *md = modifiers_getVirtualModifierList(ob, &virtualModifierData);
|
2012-03-20 22:56:26 +00:00
|
|
|
ModifierData *pretessellatePoint;
|
2012-05-06 17:22:54 +00:00
|
|
|
Curve *cu = ob->data;
|
Put the Radius property of Curve points under shape key control.
Since shape keys are stored as raw floating point data, this
unfortunately requires changes to all code that works with it.
An additional complication is that bezier and nurbs control
points have different entry size, and can be mixed in the same
object (and hence shape key buffer).
Shape key entries are changed from:
bezier: float v1[3], v2[3], v3[3], tilt, pad, pad;
nurbs: float vec[3], tilt;
To:
bezier: float v1[3], v2[3], v3[3], tilt, radius, pad;
nurbs: float vec[3], tilt, radius, pad;
The official shape key element size is changed to 3 floats,
with 4 elements for bezier nodes, and 2 for nurbs. This also
means that the element count is not equal to the vertex count
anymore.
While searching for all curve Shape Key code, I also found that
BKE_curve_transform_ex and BKE_curve_translate were broken. This
can be seen by trying to change the Origin of a Curve with keys.
Reviewers: campbellbarton, sergey
Differential Revision: https://developer.blender.org/D3676
2018-09-05 16:17:59 +03:00
|
|
|
int numElems = 0, numVerts = 0;
|
2014-03-20 22:56:28 +11:00
|
|
|
const bool editmode = (!for_render && (cu->editnurb || cu->editfont));
|
2012-05-09 15:00:26 +00:00
|
|
|
ModifierApplyFlag app_flag = 0;
|
2005-08-14 12:17:34 +00:00
|
|
|
float (*deformedVerts)[3] = NULL;
|
2012-05-06 17:22:54 +00:00
|
|
|
float *keyVerts = NULL;
|
2006-08-28 01:12:36 +00:00
|
|
|
int required_mode;
|
|
|
|
|
|
2014-02-15 17:25:56 +11:00
|
|
|
modifiers_clearErrors(ob);
|
|
|
|
|
|
2012-05-09 15:00:26 +00:00
|
|
|
if (editmode)
|
|
|
|
|
app_flag |= MOD_APPLY_USECACHE;
|
2014-03-20 22:56:28 +11:00
|
|
|
if (use_render_resolution) {
|
2012-05-09 15:00:26 +00:00
|
|
|
app_flag |= MOD_APPLY_RENDER;
|
2012-05-07 06:58:03 +00:00
|
|
|
required_mode = eModifierMode_Render;
|
2012-05-09 15:00:26 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
else
|
|
|
|
|
required_mode = eModifierMode_Realtime;
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2018-05-01 17:33:04 +02:00
|
|
|
const ModifierEvalContext mectx = {depsgraph, ob, app_flag};
|
|
|
|
|
|
2014-03-20 22:56:28 +11:00
|
|
|
pretessellatePoint = curve_get_tessellate_point(scene, ob, use_render_resolution, editmode);
|
2012-05-07 06:58:03 +00:00
|
|
|
|
|
|
|
|
if (editmode)
|
|
|
|
|
required_mode |= eModifierMode_Editmode;
|
2005-08-14 12:17:34 +00:00
|
|
|
|
Fix T50745: Shape key editing on bezier objects broken with Rendered Viewport Shading
So... Curve+shapekey was even more broken than it looked, this report was
actually a nice crasher (immediate crash in an ASAN build when trying to
edit a curve shapekey with some viewport rendering enabled).
There were actually two different issues here.
I) The less critical: rB6f1493f68fe was not fully fixing issues from
T50614. More specifically, if you updated obdata from editnurb
*without* freeing editnurb afterwards, you had a 'restored' (to
original curve) editnurb, without the edited shapekey modifications
anymore. This was fixed by tweaking again `calc_shapeKeys()` behavior in
`ED_curve_editnurb_load()`.
II) The crasher: in `ED_curve_editnurb_make()`, the call to
`init_editNurb_keyIndex()` was directly storing pointers of obdata
nurbs. Since those get freed every time `ED_curve_editnurb_load()` is
executed, it easily ended up being pointers to freed memory. This was
fixed by copying those data, which implied more complex handling code
for editnurbs->keyindex, and some reshuffling of a few functions to
avoid duplicating things between editor's editcurve.c and BKE's curve.c
Note that the separation of functions between editors and BKE area for
curve could use a serious update, it's currently messy to say the least.
Then again, that area is due to rework since a long time now... :/
Finally, aligned 'for_render' curve evaluation to mesh one - now
editing a shapekey will show in rendered viewports, if it does have some
weight (exactly as with shapekeys of meshes).
2017-02-22 21:20:50 +01:00
|
|
|
if (!editmode) {
|
Put the Radius property of Curve points under shape key control.
Since shape keys are stored as raw floating point data, this
unfortunately requires changes to all code that works with it.
An additional complication is that bezier and nurbs control
points have different entry size, and can be mixed in the same
object (and hence shape key buffer).
Shape key entries are changed from:
bezier: float v1[3], v2[3], v3[3], tilt, pad, pad;
nurbs: float vec[3], tilt;
To:
bezier: float v1[3], v2[3], v3[3], tilt, radius, pad;
nurbs: float vec[3], tilt, radius, pad;
The official shape key element size is changed to 3 floats,
with 4 elements for bezier nodes, and 2 for nurbs. This also
means that the element count is not equal to the vertex count
anymore.
While searching for all curve Shape Key code, I also found that
BKE_curve_transform_ex and BKE_curve_translate were broken. This
can be seen by trying to change the Origin of a Curve with keys.
Reviewers: campbellbarton, sergey
Differential Revision: https://developer.blender.org/D3676
2018-09-05 16:17:59 +03:00
|
|
|
keyVerts = BKE_key_evaluate_object(ob, &numElems);
|
2009-10-22 09:31:07 +00:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (keyVerts) {
|
Put the Radius property of Curve points under shape key control.
Since shape keys are stored as raw floating point data, this
unfortunately requires changes to all code that works with it.
An additional complication is that bezier and nurbs control
points have different entry size, and can be mixed in the same
object (and hence shape key buffer).
Shape key entries are changed from:
bezier: float v1[3], v2[3], v3[3], tilt, pad, pad;
nurbs: float vec[3], tilt;
To:
bezier: float v1[3], v2[3], v3[3], tilt, radius, pad;
nurbs: float vec[3], tilt, radius, pad;
The official shape key element size is changed to 3 floats,
with 4 elements for bezier nodes, and 2 for nurbs. This also
means that the element count is not equal to the vertex count
anymore.
While searching for all curve Shape Key code, I also found that
BKE_curve_transform_ex and BKE_curve_translate were broken. This
can be seen by trying to change the Origin of a Curve with keys.
Reviewers: campbellbarton, sergey
Differential Revision: https://developer.blender.org/D3676
2018-09-05 16:17:59 +03:00
|
|
|
BLI_assert(BKE_keyblock_curve_element_count(nurb) == numElems);
|
|
|
|
|
|
2009-10-22 09:31:07 +00:00
|
|
|
/* split coords from key data, the latter also includes
|
2012-03-09 18:28:30 +00:00
|
|
|
* tilts, which is passed through in the modifier stack.
|
|
|
|
|
* this is also the reason curves do not use a virtual
|
|
|
|
|
* shape key modifier yet. */
|
2013-08-19 09:29:51 +00:00
|
|
|
deformedVerts = BKE_curve_nurbs_keyVertexCos_get(nurb, keyVerts);
|
Put the Radius property of Curve points under shape key control.
Since shape keys are stored as raw floating point data, this
unfortunately requires changes to all code that works with it.
An additional complication is that bezier and nurbs control
points have different entry size, and can be mixed in the same
object (and hence shape key buffer).
Shape key entries are changed from:
bezier: float v1[3], v2[3], v3[3], tilt, pad, pad;
nurbs: float vec[3], tilt;
To:
bezier: float v1[3], v2[3], v3[3], tilt, radius, pad;
nurbs: float vec[3], tilt, radius, pad;
The official shape key element size is changed to 3 floats,
with 4 elements for bezier nodes, and 2 for nurbs. This also
means that the element count is not equal to the vertex count
anymore.
While searching for all curve Shape Key code, I also found that
BKE_curve_transform_ex and BKE_curve_translate were broken. This
can be seen by trying to change the Origin of a Curve with keys.
Reviewers: campbellbarton, sergey
Differential Revision: https://developer.blender.org/D3676
2018-09-05 16:17:59 +03:00
|
|
|
numVerts = BKE_nurbList_verts_count(nurb);
|
2009-10-22 09:31:07 +00:00
|
|
|
}
|
2005-09-26 15:34:21 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-03-20 22:56:26 +00:00
|
|
|
if (pretessellatePoint) {
|
2012-05-06 17:22:54 +00:00
|
|
|
for (; md; md = md->next) {
|
2015-03-30 21:17:07 +11:00
|
|
|
const ModifierTypeInfo *mti = modifierType_getInfo(md->type);
|
2005-08-14 12:17:34 +00:00
|
|
|
|
2012-05-17 13:22:19 +00:00
|
|
|
if (!modifier_isEnabled(scene, md, required_mode))
|
2012-05-07 06:58:03 +00:00
|
|
|
continue;
|
|
|
|
|
if (mti->type != eModifierTypeType_OnlyDeform)
|
|
|
|
|
continue;
|
2005-08-14 12:17:34 +00:00
|
|
|
|
|
|
|
|
if (!deformedVerts) {
|
2013-08-19 09:29:51 +00:00
|
|
|
deformedVerts = BKE_curve_nurbs_vertexCos_get(nurb, &numVerts);
|
2005-08-14 12:17:34 +00:00
|
|
|
}
|
2010-03-08 13:49:13 +00:00
|
|
|
|
2018-09-19 16:59:05 +02:00
|
|
|
mti->deformVerts(md, &mectx, NULL, deformedVerts, numVerts);
|
2005-08-14 12:17:34 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
if (md == pretessellatePoint)
|
2005-08-14 12:17:34 +00:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-08-19 09:36:40 +00:00
|
|
|
if (deformedVerts) {
|
2013-08-19 09:29:51 +00:00
|
|
|
BK_curve_nurbs_vertexCos_apply(nurb, deformedVerts);
|
2013-08-19 09:36:40 +00:00
|
|
|
MEM_freeN(deformedVerts);
|
|
|
|
|
}
|
2009-10-22 09:31:07 +00:00
|
|
|
if (keyVerts) /* these are not passed through modifier stack */
|
2013-08-19 09:29:51 +00:00
|
|
|
BKE_curve_nurbs_keyVertexTilts_apply(nurb, keyVerts);
|
2010-03-08 13:49:13 +00:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (keyVerts)
|
2009-10-22 09:31:07 +00:00
|
|
|
MEM_freeN(keyVerts);
|
2005-08-14 12:17:34 +00:00
|
|
|
}
|
|
|
|
|
|
2013-03-26 07:29:01 +00:00
|
|
|
static float (*displist_get_allverts(ListBase *dispbase, int *totvert))[3]
|
2010-05-13 19:23:52 +00:00
|
|
|
{
|
|
|
|
|
DispList *dl;
|
|
|
|
|
float (*allverts)[3], *fp;
|
|
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
*totvert = 0;
|
2010-05-13 19:23:52 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
for (dl = dispbase->first; dl; dl = dl->next)
|
|
|
|
|
*totvert += (dl->type == DL_INDEX3) ? dl->nr : dl->parts * dl->nr;
|
2010-05-13 19:23:52 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
allverts = MEM_mallocN((*totvert) * sizeof(float) * 3, "displist_get_allverts allverts");
|
|
|
|
|
fp = (float *)allverts;
|
|
|
|
|
for (dl = dispbase->first; dl; dl = dl->next) {
|
|
|
|
|
int offs = 3 * ((dl->type == DL_INDEX3) ? dl->nr : dl->parts * dl->nr);
|
2010-05-13 19:23:52 +00:00
|
|
|
memcpy(fp, dl->verts, sizeof(float) * offs);
|
2012-05-06 17:22:54 +00:00
|
|
|
fp += offs;
|
2010-05-13 19:23:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return allverts;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void displist_apply_allverts(ListBase *dispbase, float (*allverts)[3])
|
|
|
|
|
{
|
|
|
|
|
DispList *dl;
|
2014-04-27 00:20:13 +10:00
|
|
|
const float *fp;
|
2010-05-13 19:23:52 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
fp = (float *)allverts;
|
|
|
|
|
for (dl = dispbase->first; dl; dl = dl->next) {
|
|
|
|
|
int offs = 3 * ((dl->type == DL_INDEX3) ? dl->nr : dl->parts * dl->nr);
|
2010-05-13 19:23:52 +00:00
|
|
|
memcpy(dl->verts, fp, sizeof(float) * offs);
|
2012-05-06 17:22:54 +00:00
|
|
|
fp += offs;
|
2010-05-13 19:23:52 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-16 12:45:11 +10:00
|
|
|
static void curve_calc_modifiers_post(
|
2018-04-06 12:07:27 +02:00
|
|
|
Depsgraph *depsgraph, Scene *scene, Object *ob, ListBase *nurb,
|
2018-10-15 15:58:58 +11:00
|
|
|
ListBase *dispbase, Mesh **r_final,
|
2017-08-16 12:45:11 +10:00
|
|
|
const bool for_render, const bool use_render_resolution)
|
2005-08-14 12:17:34 +00:00
|
|
|
{
|
2013-08-19 09:05:34 +00:00
|
|
|
VirtualModifierData virtualModifierData;
|
|
|
|
|
ModifierData *md = modifiers_getVirtualModifierList(ob, &virtualModifierData);
|
2012-03-20 22:56:26 +00:00
|
|
|
ModifierData *pretessellatePoint;
|
2012-05-06 17:22:54 +00:00
|
|
|
Curve *cu = ob->data;
|
2010-07-25 11:57:36 +00:00
|
|
|
int required_mode = 0, totvert = 0;
|
2014-03-20 22:56:28 +11:00
|
|
|
const bool editmode = (!for_render && (cu->editnurb || cu->editfont));
|
2018-06-05 15:59:30 +02:00
|
|
|
Mesh *modified = NULL, *mesh_applied;
|
2010-05-13 19:23:52 +00:00
|
|
|
float (*vertCos)[3] = NULL;
|
2014-03-20 22:56:28 +11:00
|
|
|
int useCache = !for_render;
|
2012-05-09 15:00:26 +00:00
|
|
|
ModifierApplyFlag app_flag = 0;
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2014-03-20 22:56:28 +11:00
|
|
|
if (use_render_resolution) {
|
2012-05-09 15:00:26 +00:00
|
|
|
app_flag |= MOD_APPLY_RENDER;
|
2012-05-07 06:58:03 +00:00
|
|
|
required_mode = eModifierMode_Render;
|
2012-05-09 15:00:26 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
else
|
|
|
|
|
required_mode = eModifierMode_Realtime;
|
2006-08-28 01:12:36 +00:00
|
|
|
|
2018-05-01 17:33:04 +02:00
|
|
|
const ModifierEvalContext mectx_deform = {depsgraph, ob,
|
|
|
|
|
editmode ? app_flag | MOD_APPLY_USECACHE : app_flag};
|
|
|
|
|
const ModifierEvalContext mectx_apply = {depsgraph, ob,
|
|
|
|
|
useCache ? app_flag | MOD_APPLY_USECACHE : app_flag};
|
|
|
|
|
|
2014-03-20 22:56:28 +11:00
|
|
|
pretessellatePoint = curve_get_tessellate_point(scene, ob, use_render_resolution, editmode);
|
2012-05-07 06:58:03 +00:00
|
|
|
|
|
|
|
|
if (editmode)
|
|
|
|
|
required_mode |= eModifierMode_Editmode;
|
2005-08-14 12:17:34 +00:00
|
|
|
|
2012-03-20 22:56:26 +00:00
|
|
|
if (pretessellatePoint) {
|
|
|
|
|
md = pretessellatePoint->next;
|
2005-08-14 12:17:34 +00:00
|
|
|
}
|
|
|
|
|
|
2018-10-15 15:58:58 +11:00
|
|
|
if (r_final && *r_final) {
|
|
|
|
|
BKE_id_free(NULL, r_final);
|
2010-03-05 16:47:52 +00:00
|
|
|
}
|
|
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
for (; md; md = md->next) {
|
2015-03-30 21:17:07 +11:00
|
|
|
const ModifierTypeInfo *mti = modifierType_getInfo(md->type);
|
2010-03-07 10:40:52 +00:00
|
|
|
|
2012-05-17 13:22:19 +00:00
|
|
|
if (!modifier_isEnabled(scene, md, required_mode))
|
2012-05-07 06:58:03 +00:00
|
|
|
continue;
|
2005-08-14 12:17:34 +00:00
|
|
|
|
2018-12-27 15:01:48 +01:00
|
|
|
/* If we need normals, no choice, have to convert to mesh now. */
|
|
|
|
|
if (mti->dependsOnNormals != NULL && mti->dependsOnNormals(md) && modified == NULL) {
|
|
|
|
|
if (vertCos != NULL) {
|
|
|
|
|
displist_apply_allverts(dispbase, vertCos);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ELEM(ob->type, OB_CURVE, OB_FONT) && (cu->flag & CU_DEFORM_FILL)) {
|
|
|
|
|
curve_to_filledpoly(cu, nurb, dispbase);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
modified = BKE_mesh_new_nomain_from_curve_displist(ob, dispbase);
|
|
|
|
|
}
|
|
|
|
|
|
2010-05-13 19:23:52 +00:00
|
|
|
if (mti->type == eModifierTypeType_OnlyDeform ||
|
2018-06-05 15:59:30 +02:00
|
|
|
(mti->type == eModifierTypeType_DeformOrConstruct && !modified))
|
2012-05-20 19:49:27 +00:00
|
|
|
{
|
2018-06-05 15:59:30 +02:00
|
|
|
if (modified) {
|
2010-05-13 19:23:52 +00:00
|
|
|
if (!vertCos) {
|
2018-06-05 15:59:30 +02:00
|
|
|
vertCos = BKE_mesh_vertexCos_get(modified, &totvert);
|
2010-03-08 10:05:51 +00:00
|
|
|
}
|
2018-12-27 15:01:48 +01:00
|
|
|
if (mti->dependsOnNormals != NULL && mti->dependsOnNormals(md)) {
|
|
|
|
|
BKE_mesh_ensure_normals(modified);
|
|
|
|
|
}
|
2018-09-19 16:59:05 +02:00
|
|
|
mti->deformVerts(md, &mectx_deform, modified, vertCos, totvert);
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
|
|
|
|
else {
|
2010-05-13 19:23:52 +00:00
|
|
|
if (!vertCos) {
|
2012-05-06 17:22:54 +00:00
|
|
|
vertCos = displist_get_allverts(dispbase, &totvert);
|
2010-03-05 16:47:52 +00:00
|
|
|
}
|
2018-09-19 16:59:05 +02:00
|
|
|
mti->deformVerts(md, &mectx_deform, NULL, vertCos, totvert);
|
2008-10-04 12:31:54 +00:00
|
|
|
}
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
|
|
|
|
else {
|
2018-10-15 15:58:58 +11:00
|
|
|
if (!r_final) {
|
2012-05-07 06:58:03 +00:00
|
|
|
/* makeDisplistCurveTypes could be used for beveling, where derived mesh
|
|
|
|
|
* is totally unnecessary, so we could stop modifiers applying
|
|
|
|
|
* when we found constructive modifier but derived mesh is unwanted result
|
|
|
|
|
*/
|
2010-03-26 15:06:30 +00:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-05 15:59:30 +02:00
|
|
|
if (modified) {
|
2010-05-13 19:23:52 +00:00
|
|
|
if (vertCos) {
|
2018-06-05 15:59:30 +02:00
|
|
|
Mesh *temp_mesh;
|
2019-02-04 17:51:15 +01:00
|
|
|
BKE_id_copy_ex(NULL, &modified->id, (ID **)&temp_mesh, LIB_ID_COPY_LOCALIZE);
|
2018-06-05 15:59:30 +02:00
|
|
|
BKE_id_free(NULL, modified);
|
|
|
|
|
modified = temp_mesh;
|
|
|
|
|
|
|
|
|
|
BKE_mesh_apply_vert_coords(modified, vertCos);
|
2010-03-08 10:05:51 +00:00
|
|
|
}
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
|
|
|
|
else {
|
2010-05-13 19:23:52 +00:00
|
|
|
if (vertCos) {
|
|
|
|
|
displist_apply_allverts(dispbase, vertCos);
|
|
|
|
|
}
|
|
|
|
|
|
2010-03-16 21:09:53 +00:00
|
|
|
if (ELEM(ob->type, OB_CURVE, OB_FONT) && (cu->flag & CU_DEFORM_FILL)) {
|
2010-03-25 18:41:57 +00:00
|
|
|
curve_to_filledpoly(cu, nurb, dispbase);
|
2010-03-16 21:09:53 +00:00
|
|
|
}
|
|
|
|
|
|
2018-06-05 15:59:30 +02:00
|
|
|
modified = BKE_mesh_new_nomain_from_curve_displist(ob, dispbase);
|
2010-03-08 10:05:51 +00:00
|
|
|
}
|
|
|
|
|
|
2010-05-13 19:23:52 +00:00
|
|
|
if (vertCos) {
|
|
|
|
|
/* Vertex coordinates were applied to necessary data, could free it */
|
|
|
|
|
MEM_freeN(vertCos);
|
2012-05-06 17:22:54 +00:00
|
|
|
vertCos = NULL;
|
2010-05-13 19:23:52 +00:00
|
|
|
}
|
|
|
|
|
|
2018-12-27 15:01:48 +01:00
|
|
|
if (mti->dependsOnNormals != NULL && mti->dependsOnNormals(md)) {
|
|
|
|
|
BKE_mesh_ensure_normals(modified);
|
|
|
|
|
}
|
2018-09-20 12:04:17 +02:00
|
|
|
mesh_applied = mti->applyModifier(md, &mectx_apply, modified);
|
2010-03-08 10:05:51 +00:00
|
|
|
|
2018-06-05 15:59:30 +02:00
|
|
|
if (mesh_applied) {
|
2010-03-08 10:05:51 +00:00
|
|
|
/* Modifier returned a new derived mesh */
|
|
|
|
|
|
2018-06-05 15:59:30 +02:00
|
|
|
if (modified && modified != mesh_applied) /* Modifier */
|
|
|
|
|
BKE_id_free(NULL, modified);
|
|
|
|
|
modified = mesh_applied;
|
2010-03-08 10:05:51 +00:00
|
|
|
}
|
2005-08-14 12:17:34 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-05-13 19:23:52 +00:00
|
|
|
if (vertCos) {
|
2018-06-05 15:59:30 +02:00
|
|
|
if (modified) {
|
|
|
|
|
Mesh *temp_mesh;
|
2019-02-04 17:51:15 +01:00
|
|
|
BKE_id_copy_ex(NULL, &modified->id, (ID **)&temp_mesh, LIB_ID_COPY_LOCALIZE);
|
2018-06-05 15:59:30 +02:00
|
|
|
BKE_id_free(NULL, modified);
|
|
|
|
|
modified = temp_mesh;
|
|
|
|
|
|
|
|
|
|
BKE_mesh_apply_vert_coords(modified, vertCos);
|
|
|
|
|
BKE_mesh_calc_normals_mapping_simple(modified);
|
2010-03-08 10:05:51 +00:00
|
|
|
|
2010-05-13 19:23:52 +00:00
|
|
|
MEM_freeN(vertCos);
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
|
|
|
|
else {
|
2010-05-13 19:23:52 +00:00
|
|
|
displist_apply_allverts(dispbase, vertCos);
|
|
|
|
|
MEM_freeN(vertCos);
|
2012-05-06 17:22:54 +00:00
|
|
|
vertCos = NULL;
|
2010-05-13 19:23:52 +00:00
|
|
|
}
|
2010-03-08 10:05:51 +00:00
|
|
|
}
|
|
|
|
|
|
2018-10-15 15:58:58 +11:00
|
|
|
if (r_final) {
|
2018-06-05 15:59:30 +02:00
|
|
|
if (modified) {
|
2013-06-19 08:19:36 +00:00
|
|
|
/* see: mesh_calc_modifiers */
|
2018-06-05 15:59:30 +02:00
|
|
|
if (modified->totface == 0) {
|
|
|
|
|
BKE_mesh_tessface_calc(modified);
|
2013-06-19 08:19:36 +00:00
|
|
|
}
|
|
|
|
|
/* Even if tessellation is not needed, some modifiers might have modified CD layers
|
|
|
|
|
* (like mloopcol or mloopuv), hence we have to update those. */
|
2018-06-05 15:59:30 +02:00
|
|
|
else if (modified->runtime.cd_dirty_vert & CD_MASK_TESSLOOPNORMAL) {
|
|
|
|
|
BKE_mesh_tessface_calc(modified);
|
2013-06-19 08:19:36 +00:00
|
|
|
}
|
|
|
|
|
|
2018-06-05 15:59:30 +02:00
|
|
|
/* XXX2.8(Sybren): make sure the face normals are recalculated as well */
|
|
|
|
|
BKE_mesh_ensure_normals(modified);
|
|
|
|
|
|
2019-01-03 18:01:56 +01:00
|
|
|
/* Special tweaks, needed since neither BKE_mesh_new_nomain_from_template() nor
|
|
|
|
|
* BKE_mesh_new_nomain_from_curve_displist() properly duplicate mat info...
|
|
|
|
|
*/
|
|
|
|
|
BLI_strncpy(modified->id.name, cu->id.name, sizeof(modified->id.name));
|
|
|
|
|
*((short *)modified->id.name) = ID_ME;
|
|
|
|
|
MEM_SAFE_FREE(modified->mat);
|
|
|
|
|
/* Set flag which makes it easier to see what's going on in a debugger. */
|
|
|
|
|
modified->id.tag |= LIB_TAG_COPIED_ON_WRITE_EVAL_RESULT;
|
|
|
|
|
modified->mat = MEM_dupallocN(cu->mat);
|
|
|
|
|
modified->totcol = cu->totcol;
|
|
|
|
|
|
2018-10-15 15:58:58 +11:00
|
|
|
(*r_final) = modified;
|
2013-06-19 08:19:36 +00:00
|
|
|
}
|
2018-06-05 15:59:30 +02:00
|
|
|
else {
|
2018-10-15 15:58:58 +11:00
|
|
|
(*r_final) = NULL;
|
2018-06-05 15:59:30 +02:00
|
|
|
}
|
2010-03-26 15:06:30 +00:00
|
|
|
}
|
2005-08-14 12:17:34 +00:00
|
|
|
}
|
|
|
|
|
|
2006-11-20 21:25:02 +00:00
|
|
|
static void displist_surf_indices(DispList *dl)
|
|
|
|
|
{
|
|
|
|
|
int a, b, p1, p2, p3, p4;
|
|
|
|
|
int *index;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->totindex = 0;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
index = dl->index = MEM_mallocN(4 * sizeof(int) * (dl->parts + 1) * (dl->nr + 1), "index array nurbs");
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
for (a = 0; a < dl->parts; a++) {
|
2012-05-07 06:58:03 +00:00
|
|
|
|
|
|
|
|
if (BKE_displist_surfindex_get(dl, a, &b, &p1, &p2, &p3, &p4) == 0)
|
2008-10-06 06:10:14 +00:00
|
|
|
break;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
for (; b < dl->nr; b++, index += 4) {
|
|
|
|
|
index[0] = p1;
|
|
|
|
|
index[1] = p2;
|
|
|
|
|
index[2] = p4;
|
|
|
|
|
index[3] = p3;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2006-11-20 21:25:02 +00:00
|
|
|
dl->totindex++;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
p2 = p1; p1++;
|
|
|
|
|
p4 = p3; p3++;
|
2006-11-20 21:25:02 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-05 18:31:47 +02:00
|
|
|
/* XXX2.8(Sybren): unused function; impossible to test after porting to Mesh */
|
|
|
|
|
#ifdef WITH_DERIVEDMESH_DEPRECATED_FUNCS
|
2018-04-06 12:07:27 +02:00
|
|
|
static DerivedMesh *create_orco_dm(Depsgraph *depsgraph, Scene *scene, Object *ob)
|
2010-03-05 16:47:52 +00:00
|
|
|
{
|
|
|
|
|
DerivedMesh *dm;
|
2012-05-06 17:22:54 +00:00
|
|
|
ListBase disp = {NULL, NULL};
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2010-03-08 13:49:13 +00:00
|
|
|
/* OrcoDM should be created from underformed disp lists */
|
2018-04-06 12:07:27 +02:00
|
|
|
BKE_displist_make_curveTypes_forOrco(depsgraph, scene, ob, &disp);
|
2013-01-10 17:37:17 +00:00
|
|
|
dm = CDDM_from_curve_displist(ob, &disp);
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-07 06:58:03 +00:00
|
|
|
BKE_displist_free(&disp);
|
2010-03-05 16:47:52 +00:00
|
|
|
|
|
|
|
|
return dm;
|
|
|
|
|
}
|
|
|
|
|
|
2013-11-26 01:17:24 +06:00
|
|
|
static void add_orco_dm(Object *ob, DerivedMesh *dm, DerivedMesh *orcodm)
|
2010-03-05 16:47:52 +00:00
|
|
|
{
|
|
|
|
|
float (*orco)[3], (*layerorco)[3];
|
|
|
|
|
int totvert, a;
|
2012-05-06 17:22:54 +00:00
|
|
|
Curve *cu = ob->data;
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
totvert = dm->getNumVerts(dm);
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2013-11-26 01:17:24 +06:00
|
|
|
orco = MEM_callocN(sizeof(float) * 3 * totvert, "dm orco");
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2013-11-26 01:17:24 +06:00
|
|
|
if (orcodm->getNumVerts(orcodm) == totvert)
|
|
|
|
|
orcodm->getVertCos(orcodm, orco);
|
|
|
|
|
else
|
|
|
|
|
dm->getVertCos(dm, orco);
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
for (a = 0; a < totvert; a++) {
|
2010-03-05 16:47:52 +00:00
|
|
|
float *co = orco[a];
|
2012-05-06 17:22:54 +00:00
|
|
|
co[0] = (co[0] - cu->loc[0]) / cu->size[0];
|
|
|
|
|
co[1] = (co[1] - cu->loc[1]) / cu->size[1];
|
|
|
|
|
co[2] = (co[2] - cu->loc[2]) / cu->size[2];
|
2010-03-05 16:47:52 +00:00
|
|
|
}
|
|
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if ((layerorco = DM_get_vert_data_layer(dm, CD_ORCO))) {
|
2012-05-06 17:22:54 +00:00
|
|
|
memcpy(layerorco, orco, sizeof(float) * totvert);
|
2010-03-05 16:47:52 +00:00
|
|
|
MEM_freeN(orco);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
DM_add_vert_layer(dm, CD_ORCO, CD_ASSIGN, orco);
|
|
|
|
|
}
|
2018-06-05 18:31:47 +02:00
|
|
|
#endif
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2018-06-05 18:31:47 +02:00
|
|
|
/* XXX2.8(Sybren): unused function; impossible to test after porting to Mesh */
|
|
|
|
|
#ifdef WITH_DERIVEDMESH_DEPRECATED_FUNCS
|
2017-08-16 12:45:11 +10:00
|
|
|
static void curve_calc_orcodm(
|
2018-04-06 12:07:27 +02:00
|
|
|
Depsgraph *depsgraph, Scene *scene, Object *ob, DerivedMesh *dm_final,
|
2017-08-16 12:45:11 +10:00
|
|
|
const bool for_render, const bool use_render_resolution)
|
2010-03-05 16:47:52 +00:00
|
|
|
{
|
2012-05-07 06:58:03 +00:00
|
|
|
/* this function represents logic of mesh's orcodm calculation
|
|
|
|
|
* for displist-based objects
|
|
|
|
|
*/
|
2013-08-19 09:05:34 +00:00
|
|
|
VirtualModifierData virtualModifierData;
|
|
|
|
|
ModifierData *md = modifiers_getVirtualModifierList(ob, &virtualModifierData);
|
2012-03-20 22:56:26 +00:00
|
|
|
ModifierData *pretessellatePoint;
|
2012-05-06 17:22:54 +00:00
|
|
|
Curve *cu = ob->data;
|
2010-03-05 16:47:52 +00:00
|
|
|
int required_mode;
|
2014-03-20 22:56:28 +11:00
|
|
|
const bool editmode = (!for_render && (cu->editnurb || cu->editfont));
|
2012-05-06 17:22:54 +00:00
|
|
|
DerivedMesh *ndm, *orcodm = NULL;
|
2013-05-02 14:42:05 +00:00
|
|
|
ModifierApplyFlag app_flag = MOD_APPLY_ORCO;
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2014-03-20 22:56:28 +11:00
|
|
|
if (use_render_resolution) {
|
2013-05-02 14:42:05 +00:00
|
|
|
app_flag |= MOD_APPLY_RENDER;
|
2012-05-07 06:58:03 +00:00
|
|
|
required_mode = eModifierMode_Render;
|
2013-05-02 14:42:05 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
else
|
|
|
|
|
required_mode = eModifierMode_Realtime;
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2018-05-01 17:33:04 +02:00
|
|
|
const ModifierEvalContext mectx = {depsgraph, ob, app_flag};
|
|
|
|
|
|
2014-03-20 22:56:28 +11:00
|
|
|
pretessellatePoint = curve_get_tessellate_point(scene, ob, use_render_resolution, editmode);
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-07 06:58:03 +00:00
|
|
|
if (editmode)
|
|
|
|
|
required_mode |= eModifierMode_Editmode;
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-03-20 22:56:26 +00:00
|
|
|
if (pretessellatePoint) {
|
|
|
|
|
md = pretessellatePoint->next;
|
2010-03-05 16:47:52 +00:00
|
|
|
}
|
|
|
|
|
|
2013-11-26 01:17:24 +06:00
|
|
|
/* If modifiers are disabled, we wouldn't be here because
|
|
|
|
|
* this function is only called if there're enabled constructive
|
|
|
|
|
* modifiers applied on the curve.
|
|
|
|
|
*
|
|
|
|
|
* This means we can create ORCO DM in advance and assume it's
|
|
|
|
|
* never NULL.
|
|
|
|
|
*/
|
2018-04-06 12:07:27 +02:00
|
|
|
orcodm = create_orco_dm(depsgraph, scene, ob);
|
2013-11-26 01:17:24 +06:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
for (; md; md = md->next) {
|
2015-03-30 21:17:07 +11:00
|
|
|
const ModifierTypeInfo *mti = modifierType_getInfo(md->type);
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
md->scene = scene;
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-05-17 13:22:19 +00:00
|
|
|
if (!modifier_isEnabled(scene, md, required_mode))
|
2012-05-07 06:58:03 +00:00
|
|
|
continue;
|
|
|
|
|
if (mti->type != eModifierTypeType_Constructive)
|
|
|
|
|
continue;
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2018-05-01 17:33:04 +02:00
|
|
|
ndm = modwrap_applyModifier(md, &mectx, orcodm);
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (ndm) {
|
2010-03-05 16:47:52 +00:00
|
|
|
/* if the modifier returned a new dm, release the old one */
|
2012-03-24 06:18:31 +00:00
|
|
|
if (orcodm && orcodm != ndm) {
|
2010-03-05 16:47:52 +00:00
|
|
|
orcodm->release(orcodm);
|
|
|
|
|
}
|
|
|
|
|
orcodm = ndm;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* add an orco layer if needed */
|
2014-03-20 22:56:28 +11:00
|
|
|
add_orco_dm(ob, dm_final, orcodm);
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2013-12-03 09:22:29 +11:00
|
|
|
orcodm->release(orcodm);
|
2010-03-05 16:47:52 +00:00
|
|
|
}
|
2018-06-05 18:31:47 +02:00
|
|
|
#endif
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2017-08-16 12:45:11 +10:00
|
|
|
void BKE_displist_make_surf(
|
2018-04-06 12:07:27 +02:00
|
|
|
Depsgraph *depsgraph, Scene *scene, Object *ob, ListBase *dispbase,
|
2018-10-15 15:58:58 +11:00
|
|
|
Mesh **r_final,
|
2017-08-16 12:45:11 +10:00
|
|
|
const bool for_render, const bool for_orco, const bool use_render_resolution)
|
2005-07-14 15:48:01 +00:00
|
|
|
{
|
2013-08-19 09:36:40 +00:00
|
|
|
ListBase nubase = {NULL, NULL};
|
2005-07-14 15:48:01 +00:00
|
|
|
Nurb *nu;
|
2005-08-14 06:08:41 +00:00
|
|
|
Curve *cu = ob->data;
|
|
|
|
|
DispList *dl;
|
|
|
|
|
float *data;
|
|
|
|
|
int len;
|
2010-07-25 11:57:36 +00:00
|
|
|
|
2014-03-20 22:56:28 +11:00
|
|
|
if (!for_render && cu->editnurb) {
|
2013-08-19 09:36:40 +00:00
|
|
|
BKE_nurbList_duplicate(&nubase, BKE_curve_editNurbs_get(cu));
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
BKE_nurbList_duplicate(&nubase, &cu->nurb);
|
|
|
|
|
}
|
2005-08-14 12:17:34 +00:00
|
|
|
|
2014-03-20 22:56:28 +11:00
|
|
|
if (!for_orco)
|
2018-04-06 12:07:27 +02:00
|
|
|
curve_calc_modifiers_pre(depsgraph, scene, ob, &nubase, for_render, use_render_resolution);
|
2005-08-14 12:17:34 +00:00
|
|
|
|
2013-08-19 09:36:40 +00:00
|
|
|
for (nu = nubase.first; nu; nu = nu->next) {
|
2014-04-08 07:28:14 +10:00
|
|
|
if ((for_render || nu->hide == 0) && BKE_nurb_check_valid_uv(nu)) {
|
2012-05-06 17:22:54 +00:00
|
|
|
int resolu = nu->resolu, resolv = nu->resolv;
|
2010-11-11 09:56:39 +00:00
|
|
|
|
2014-03-20 22:56:28 +11:00
|
|
|
if (use_render_resolution) {
|
2012-05-07 06:58:03 +00:00
|
|
|
if (cu->resolu_ren)
|
|
|
|
|
resolu = cu->resolu_ren;
|
|
|
|
|
if (cu->resolv_ren)
|
|
|
|
|
resolv = cu->resolv_ren;
|
2010-11-11 09:56:39 +00:00
|
|
|
}
|
|
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
if (nu->pntsv == 1) {
|
|
|
|
|
len = SEGMENTSU(nu) * resolu;
|
2010-03-30 09:57:58 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = MEM_callocN(sizeof(DispList), "makeDispListsurf");
|
2014-06-19 15:13:15 +10:00
|
|
|
dl->verts = MEM_mallocN(len * sizeof(float[3]), "dlverts");
|
2010-03-30 09:57:58 +00:00
|
|
|
|
2005-08-14 06:08:41 +00:00
|
|
|
BLI_addtail(dispbase, dl);
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->parts = 1;
|
|
|
|
|
dl->nr = len;
|
|
|
|
|
dl->col = nu->mat_nr;
|
|
|
|
|
dl->charidx = nu->charidx;
|
2010-03-30 09:57:58 +00:00
|
|
|
|
|
|
|
|
/* dl->rt will be used as flag for render face and */
|
|
|
|
|
/* CU_2D conflicts with R_NOPUNOFLIP */
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->rt = nu->flag & ~CU_2D;
|
2010-03-30 09:57:58 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
data = dl->verts;
|
|
|
|
|
if (nu->flagu & CU_NURB_CYCLIC) dl->type = DL_POLY;
|
|
|
|
|
else dl->type = DL_SEGM;
|
2010-03-30 09:57:58 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
BKE_nurb_makeCurve(nu, data, NULL, NULL, NULL, resolu, 3 * sizeof(float));
|
2005-08-14 06:08:41 +00:00
|
|
|
}
|
|
|
|
|
else {
|
2012-05-06 17:22:54 +00:00
|
|
|
len = (nu->pntsu * resolu) * (nu->pntsv * resolv);
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = MEM_callocN(sizeof(DispList), "makeDispListsurf");
|
2014-06-15 14:36:33 +10:00
|
|
|
dl->verts = MEM_mallocN(len * sizeof(float[3]), "dlverts");
|
2005-08-14 06:08:41 +00:00
|
|
|
BLI_addtail(dispbase, dl);
|
2005-08-11 22:27:53 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->col = nu->mat_nr;
|
|
|
|
|
dl->charidx = nu->charidx;
|
2010-03-30 09:57:58 +00:00
|
|
|
|
|
|
|
|
/* dl->rt will be used as flag for render face and */
|
|
|
|
|
/* CU_2D conflicts with R_NOPUNOFLIP */
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->rt = nu->flag & ~CU_2D;
|
2010-03-30 09:57:58 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
data = dl->verts;
|
|
|
|
|
dl->type = DL_SURF;
|
2010-03-30 09:57:58 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->parts = (nu->pntsu * resolu); /* in reverse, because makeNurbfaces works that way */
|
|
|
|
|
dl->nr = (nu->pntsv * resolv);
|
|
|
|
|
if (nu->flagv & CU_NURB_CYCLIC) dl->flag |= DL_CYCL_U; /* reverse too! */
|
|
|
|
|
if (nu->flagu & CU_NURB_CYCLIC) dl->flag |= DL_CYCL_V;
|
2005-08-11 22:27:53 +00:00
|
|
|
|
2012-04-28 16:49:00 +00:00
|
|
|
BKE_nurb_makeFaces(nu, data, 0, resolu, resolv);
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2006-11-20 21:25:02 +00:00
|
|
|
/* gl array drawing: using indices */
|
|
|
|
|
displist_surf_indices(dl);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|
2005-08-14 06:08:41 +00:00
|
|
|
}
|
|
|
|
|
|
2014-03-20 22:56:28 +11:00
|
|
|
if (!for_orco) {
|
2018-07-30 16:54:40 +02:00
|
|
|
BKE_nurbList_duplicate(&ob->runtime.curve_cache->deformed_nurbs, &nubase);
|
2018-10-15 15:58:58 +11:00
|
|
|
curve_calc_modifiers_post(depsgraph, scene, ob, &nubase, dispbase, r_final,
|
2014-03-20 22:56:28 +11:00
|
|
|
for_render, use_render_resolution);
|
2012-05-07 06:58:03 +00:00
|
|
|
}
|
2013-08-19 09:36:40 +00:00
|
|
|
|
|
|
|
|
BKE_nurbList_free(&nubase);
|
2005-08-14 06:08:41 +00:00
|
|
|
}
|
2005-10-03 14:13:47 +00:00
|
|
|
|
2014-03-16 03:24:05 +11:00
|
|
|
static void rotateBevelPiece(Curve *cu, BevPoint *bevp, BevPoint *nbevp, DispList *dlb, float bev_blend, float widfac, float fac, float **r_data)
|
2012-02-02 15:15:52 +00:00
|
|
|
{
|
2014-03-16 03:24:05 +11:00
|
|
|
float *fp, *data = *r_data;
|
2012-02-02 15:15:52 +00:00
|
|
|
int b;
|
|
|
|
|
|
|
|
|
|
fp = dlb->verts;
|
2012-05-06 17:22:54 +00:00
|
|
|
for (b = 0; b < dlb->nr; b++, fp += 3, data += 3) {
|
2012-03-24 06:18:31 +00:00
|
|
|
if (cu->flag & CU_3D) {
|
2012-11-08 08:16:44 +00:00
|
|
|
float vec[3], quat[4];
|
2012-02-02 15:15:52 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
vec[0] = fp[1] + widfac;
|
2012-02-02 15:15:52 +00:00
|
|
|
vec[1] = fp[2];
|
2012-05-06 17:22:54 +00:00
|
|
|
vec[2] = 0.0;
|
2012-02-02 15:15:52 +00:00
|
|
|
|
2012-11-08 08:16:44 +00:00
|
|
|
if (nbevp == NULL) {
|
|
|
|
|
copy_v3_v3(data, bevp->vec);
|
|
|
|
|
copy_qt_qt(quat, bevp->quat);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
interp_v3_v3v3(data, bevp->vec, nbevp->vec, bev_blend);
|
|
|
|
|
interp_qt_qtqt(quat, bevp->quat, nbevp->quat, bev_blend);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mul_qt_v3(quat, vec);
|
2012-02-02 15:15:52 +00:00
|
|
|
|
2012-11-08 08:16:44 +00:00
|
|
|
data[0] += fac * vec[0];
|
|
|
|
|
data[1] += fac * vec[1];
|
|
|
|
|
data[2] += fac * vec[2];
|
2012-02-02 15:15:52 +00:00
|
|
|
}
|
|
|
|
|
else {
|
2012-11-08 08:16:44 +00:00
|
|
|
float sina, cosa;
|
|
|
|
|
|
|
|
|
|
if (nbevp == NULL) {
|
|
|
|
|
copy_v3_v3(data, bevp->vec);
|
|
|
|
|
sina = bevp->sina;
|
|
|
|
|
cosa = bevp->cosa;
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
interp_v3_v3v3(data, bevp->vec, nbevp->vec, bev_blend);
|
|
|
|
|
|
|
|
|
|
/* perhaps we need to interpolate angles instead. but the thing is
|
|
|
|
|
* cosa and sina are not actually sine and cosine
|
|
|
|
|
*/
|
|
|
|
|
sina = nbevp->sina * bev_blend + bevp->sina * (1.0f - bev_blend);
|
|
|
|
|
cosa = nbevp->cosa * bev_blend + bevp->cosa * (1.0f - bev_blend);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
data[0] += fac * (widfac + fp[1]) * sina;
|
|
|
|
|
data[1] += fac * (widfac + fp[1]) * cosa;
|
|
|
|
|
data[2] += fac * fp[2];
|
2012-02-02 15:15:52 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-03-16 03:24:05 +11:00
|
|
|
*r_data = data;
|
2012-02-02 15:15:52 +00:00
|
|
|
}
|
|
|
|
|
|
2012-05-04 17:04:20 +00:00
|
|
|
static void fillBevelCap(Nurb *nu, DispList *dlb, float *prev_fp, ListBase *dispbase)
|
2012-02-02 15:15:52 +00:00
|
|
|
{
|
|
|
|
|
DispList *dl;
|
|
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = MEM_callocN(sizeof(DispList), "makeDispListbev2");
|
2014-06-15 14:36:33 +10:00
|
|
|
dl->verts = MEM_mallocN(sizeof(float[3]) * dlb->nr, "dlverts");
|
2013-04-26 20:04:58 +00:00
|
|
|
memcpy(dl->verts, prev_fp, 3 * sizeof(float) * dlb->nr);
|
2012-02-02 15:15:52 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->type = DL_POLY;
|
2012-02-02 15:15:52 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->parts = 1;
|
|
|
|
|
dl->nr = dlb->nr;
|
|
|
|
|
dl->col = nu->mat_nr;
|
|
|
|
|
dl->charidx = nu->charidx;
|
2012-02-02 15:15:52 +00:00
|
|
|
|
|
|
|
|
/* dl->rt will be used as flag for render face and */
|
|
|
|
|
/* CU_2D conflicts with R_NOPUNOFLIP */
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->rt = nu->flag & ~CU_2D;
|
2012-02-02 15:15:52 +00:00
|
|
|
|
2012-02-02 18:50:44 +00:00
|
|
|
BLI_addtail(dispbase, dl);
|
2012-02-02 15:15:52 +00:00
|
|
|
}
|
|
|
|
|
|
2014-08-18 17:35:51 +06:00
|
|
|
static void calc_bevfac_segment_mapping(BevList *bl, float bevfac, float spline_length, int *r_bev, float *r_blend)
|
|
|
|
|
{
|
|
|
|
|
float normlen, normsum = 0.0f;
|
|
|
|
|
float *seglen = bl->seglen;
|
|
|
|
|
int *segbevcount = bl->segbevcount;
|
|
|
|
|
int bevcount = 0, nr = bl->nr;
|
|
|
|
|
|
|
|
|
|
float bev_fl = bevfac * (bl->nr - 1);
|
|
|
|
|
*r_bev = (int)bev_fl;
|
|
|
|
|
|
|
|
|
|
while (bevcount < nr - 1) {
|
|
|
|
|
normlen = *seglen / spline_length;
|
2014-08-20 08:47:32 +10:00
|
|
|
if (normsum + normlen > bevfac) {
|
2014-08-18 17:35:51 +06:00
|
|
|
bev_fl = bevcount + (bevfac - normsum) / normlen * *segbevcount;
|
|
|
|
|
*r_bev = (int) bev_fl;
|
|
|
|
|
*r_blend = bev_fl - *r_bev;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
normsum += normlen;
|
|
|
|
|
bevcount += *segbevcount;
|
|
|
|
|
segbevcount++;
|
|
|
|
|
seglen++;
|
|
|
|
|
}
|
|
|
|
|
}
|
2014-03-28 16:41:56 +06:00
|
|
|
|
2014-08-18 17:35:51 +06:00
|
|
|
static void calc_bevfac_spline_mapping(BevList *bl, float bevfac,
|
|
|
|
|
float spline_length,
|
2014-03-28 16:41:56 +06:00
|
|
|
int *r_bev, float *r_blend)
|
|
|
|
|
{
|
2014-05-26 21:57:53 +10:00
|
|
|
const float len_target = bevfac * spline_length;
|
2014-08-18 17:35:51 +06:00
|
|
|
BevPoint *bevp = bl->bevpoints;
|
|
|
|
|
float len_next = 0.0f, len = 0.0f;
|
|
|
|
|
int i = 0, nr = bl->nr;
|
|
|
|
|
|
|
|
|
|
while (nr--) {
|
|
|
|
|
bevp++;
|
|
|
|
|
len_next = len + bevp->offset;
|
2014-05-26 21:57:53 +10:00
|
|
|
if (len_next > len_target) {
|
2014-03-28 16:41:56 +06:00
|
|
|
break;
|
|
|
|
|
}
|
2014-05-26 21:57:53 +10:00
|
|
|
len = len_next;
|
2014-08-18 17:35:51 +06:00
|
|
|
i++;
|
2014-03-28 16:41:56 +06:00
|
|
|
}
|
2014-05-26 21:57:53 +10:00
|
|
|
|
|
|
|
|
*r_bev = i;
|
2014-08-18 17:35:51 +06:00
|
|
|
*r_blend = (len_target - len) / bevp->offset;
|
2014-03-28 16:41:56 +06:00
|
|
|
}
|
|
|
|
|
|
2014-08-18 17:35:51 +06:00
|
|
|
static void calc_bevfac_mapping_default(BevList *bl,
|
|
|
|
|
int *r_start, float *r_firstblend,
|
|
|
|
|
int *r_steps, float *r_lastblend)
|
2014-05-26 21:50:17 +10:00
|
|
|
{
|
|
|
|
|
*r_start = 0;
|
|
|
|
|
*r_steps = bl->nr;
|
|
|
|
|
*r_firstblend = 1.0f;
|
|
|
|
|
*r_lastblend = 1.0f;
|
|
|
|
|
}
|
|
|
|
|
|
2014-08-18 17:35:51 +06:00
|
|
|
static void calc_bevfac_mapping(Curve *cu, BevList *bl, Nurb *nu,
|
2014-05-26 23:55:34 +10:00
|
|
|
int *r_start, float *r_firstblend, int *r_steps, float *r_lastblend)
|
2014-03-28 16:41:56 +06:00
|
|
|
{
|
2014-08-18 17:35:51 +06:00
|
|
|
float tmpf, total_length = 0.0f;
|
|
|
|
|
int end = 0, i;
|
2014-03-28 16:41:56 +06:00
|
|
|
|
2014-05-26 21:50:17 +10:00
|
|
|
if ((BKE_nurb_check_valid_u(nu) == false) ||
|
|
|
|
|
/* not essential, but skips unnecessary calculation */
|
|
|
|
|
(min_ff(cu->bevfac1, cu->bevfac2) == 0.0f &&
|
|
|
|
|
max_ff(cu->bevfac1, cu->bevfac2) == 1.0f))
|
|
|
|
|
{
|
|
|
|
|
calc_bevfac_mapping_default(bl, r_start, r_firstblend, r_steps, r_lastblend);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2015-03-30 10:22:12 +02:00
|
|
|
if (ELEM(cu->bevfac1_mapping, CU_BEVFAC_MAP_SEGMENT, CU_BEVFAC_MAP_SPLINE) ||
|
|
|
|
|
ELEM(cu->bevfac2_mapping, CU_BEVFAC_MAP_SEGMENT, CU_BEVFAC_MAP_SPLINE))
|
2014-03-28 16:41:56 +06:00
|
|
|
{
|
2014-08-18 17:35:51 +06:00
|
|
|
for (i = 0; i < SEGMENTSU(nu); i++) {
|
|
|
|
|
total_length += bl->seglen[i];
|
2014-03-28 16:41:56 +06:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (cu->bevfac1_mapping) {
|
|
|
|
|
case CU_BEVFAC_MAP_RESOLU:
|
|
|
|
|
{
|
|
|
|
|
const float start_fl = cu->bevfac1 * (bl->nr - 1);
|
|
|
|
|
*r_start = (int)start_fl;
|
|
|
|
|
*r_firstblend = 1.0f - (start_fl - (*r_start));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case CU_BEVFAC_MAP_SEGMENT:
|
|
|
|
|
{
|
2014-08-18 17:35:51 +06:00
|
|
|
calc_bevfac_segment_mapping(bl, cu->bevfac1, total_length, r_start, r_firstblend);
|
|
|
|
|
*r_firstblend = 1.0f - *r_firstblend;
|
2014-03-28 16:41:56 +06:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case CU_BEVFAC_MAP_SPLINE:
|
|
|
|
|
{
|
2014-08-18 17:35:51 +06:00
|
|
|
calc_bevfac_spline_mapping(bl, cu->bevfac1, total_length, r_start, r_firstblend);
|
2014-03-28 16:41:56 +06:00
|
|
|
*r_firstblend = 1.0f - *r_firstblend;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (cu->bevfac2_mapping) {
|
|
|
|
|
case CU_BEVFAC_MAP_RESOLU:
|
|
|
|
|
{
|
|
|
|
|
const float end_fl = cu->bevfac2 * (bl->nr - 1);
|
|
|
|
|
end = (int)end_fl;
|
|
|
|
|
|
|
|
|
|
*r_steps = 2 + end - *r_start;
|
|
|
|
|
*r_lastblend = end_fl - end;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case CU_BEVFAC_MAP_SEGMENT:
|
|
|
|
|
{
|
2014-08-18 17:35:51 +06:00
|
|
|
calc_bevfac_segment_mapping(bl, cu->bevfac2, total_length, &end, r_lastblend);
|
2014-03-28 16:41:56 +06:00
|
|
|
*r_steps = end - *r_start + 2;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case CU_BEVFAC_MAP_SPLINE:
|
|
|
|
|
{
|
2014-08-18 17:35:51 +06:00
|
|
|
calc_bevfac_spline_mapping(bl, cu->bevfac2, total_length, &end, r_lastblend);
|
2014-03-28 16:41:56 +06:00
|
|
|
*r_steps = end - *r_start + 2;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-26 23:55:34 +10:00
|
|
|
if (end < *r_start || (end == *r_start && *r_lastblend < 1.0f - *r_firstblend )) {
|
2014-03-28 16:41:56 +06:00
|
|
|
SWAP(int, *r_start, end);
|
|
|
|
|
tmpf = *r_lastblend;
|
|
|
|
|
*r_lastblend = 1.0f - *r_firstblend;
|
|
|
|
|
*r_firstblend = 1.0f - tmpf;
|
|
|
|
|
*r_steps = end - *r_start + 2;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (*r_start + *r_steps > bl->nr) {
|
|
|
|
|
*r_steps = bl->nr - *r_start;
|
|
|
|
|
*r_lastblend = 1.0f;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-16 12:45:11 +10:00
|
|
|
static void do_makeDispListCurveTypes(
|
2018-04-06 12:07:27 +02:00
|
|
|
Depsgraph *depsgraph, Scene *scene, Object *ob, ListBase *dispbase,
|
2019-02-14 17:21:55 +11:00
|
|
|
const bool for_render, const bool for_orco, const bool use_render_resolution,
|
|
|
|
|
LinkNode *ob_cyclic_list,
|
|
|
|
|
Mesh **r_final)
|
2005-08-14 06:08:41 +00:00
|
|
|
{
|
|
|
|
|
Curve *cu = ob->data;
|
2010-03-08 13:49:13 +00:00
|
|
|
|
2005-12-07 19:47:22 +00:00
|
|
|
/* we do allow duplis... this is only displist on curve level */
|
2014-07-20 01:30:29 +10:00
|
|
|
if (!ELEM(ob->type, OB_SURF, OB_CURVE, OB_FONT)) return;
|
2004-09-14 19:03:11 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
if (ob->type == OB_SURF) {
|
2018-10-15 15:58:58 +11:00
|
|
|
BKE_displist_make_surf(depsgraph, scene, ob, dispbase, r_final, for_render, for_orco, use_render_resolution);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2009-01-02 19:10:35 +00:00
|
|
|
else if (ELEM(ob->type, OB_CURVE, OB_FONT)) {
|
2005-10-03 20:24:10 +00:00
|
|
|
ListBase dlbev;
|
2013-08-19 09:36:40 +00:00
|
|
|
ListBase nubase = {NULL, NULL};
|
2005-10-03 14:13:47 +00:00
|
|
|
|
2018-07-30 16:54:40 +02:00
|
|
|
BKE_curve_bevelList_free(&ob->runtime.curve_cache->bev);
|
2010-03-08 13:49:13 +00:00
|
|
|
|
2015-10-07 15:02:06 +11:00
|
|
|
/* We only re-evaluate path if evaluation is not happening for orco.
|
2014-02-25 13:15:59 +06:00
|
|
|
* If the calculation happens for orco, we should never free data which
|
|
|
|
|
* was needed before and only not needed for orco calculation.
|
|
|
|
|
*/
|
2014-03-20 22:56:28 +11:00
|
|
|
if (!for_orco) {
|
2018-07-30 16:54:40 +02:00
|
|
|
if (ob->runtime.curve_cache->path) free_path(ob->runtime.curve_cache->path);
|
|
|
|
|
ob->runtime.curve_cache->path = NULL;
|
2014-02-25 13:15:59 +06:00
|
|
|
}
|
2010-03-08 13:49:13 +00:00
|
|
|
|
2014-01-05 17:04:52 +06:00
|
|
|
if (ob->type == OB_FONT) {
|
2018-06-12 15:03:51 +02:00
|
|
|
BKE_vfont_to_curve_nubase(ob, FO_EDIT, &nubase);
|
2014-01-05 17:04:52 +06:00
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
BKE_nurbList_duplicate(&nubase, BKE_curve_nurbs_get(cu));
|
|
|
|
|
}
|
2013-08-21 13:13:48 +00:00
|
|
|
|
2014-03-20 22:56:28 +11:00
|
|
|
if (!for_orco)
|
2018-04-06 12:07:27 +02:00
|
|
|
curve_calc_modifiers_pre(depsgraph, scene, ob, &nubase, for_render, use_render_resolution);
|
2005-08-14 06:08:41 +00:00
|
|
|
|
2018-11-28 18:14:40 +01:00
|
|
|
BKE_curve_bevelList_make(ob, &nubase, use_render_resolution);
|
2005-08-14 06:08:41 +00:00
|
|
|
|
2006-08-13 07:37:51 +00:00
|
|
|
/* If curve has no bevel will return nothing */
|
2019-02-14 17:21:55 +11:00
|
|
|
BKE_curve_bevel_make(
|
|
|
|
|
depsgraph, scene, ob, &dlbev, for_render, use_render_resolution,
|
|
|
|
|
ob_cyclic_list);
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2006-06-06 10:09:44 +00:00
|
|
|
/* no bevel or extrude, and no width correction? */
|
2012-05-06 17:22:54 +00:00
|
|
|
if (!dlbev.first && cu->width == 1.0f) {
|
2014-03-20 22:56:28 +11:00
|
|
|
curve_to_displist(cu, &nubase, dispbase, for_render, use_render_resolution);
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
|
|
|
|
else {
|
2012-05-06 17:22:54 +00:00
|
|
|
float widfac = cu->width - 1.0f;
|
2018-07-30 16:54:40 +02:00
|
|
|
BevList *bl = ob->runtime.curve_cache->bev.first;
|
2013-08-19 09:36:40 +00:00
|
|
|
Nurb *nu = nubase.first;
|
2005-08-14 06:08:41 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
for (; bl && nu; bl = bl->next, nu = nu->next) {
|
2006-06-06 10:09:44 +00:00
|
|
|
DispList *dl;
|
2012-02-02 15:15:52 +00:00
|
|
|
float *data;
|
|
|
|
|
int a;
|
2010-03-30 09:57:58 +00:00
|
|
|
|
2007-12-03 22:42:48 +00:00
|
|
|
if (bl->nr) { /* blank bevel lists can happen */
|
2010-03-30 09:57:58 +00:00
|
|
|
|
2007-12-03 22:42:48 +00:00
|
|
|
/* exception handling; curve without bevel or extrude, with width correction */
|
2014-02-08 06:07:10 +11:00
|
|
|
if (BLI_listbase_is_empty(&dlbev)) {
|
2014-05-26 21:39:46 +10:00
|
|
|
BevPoint *bevp;
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = MEM_callocN(sizeof(DispList), "makeDispListbev");
|
2014-06-15 14:36:33 +10:00
|
|
|
dl->verts = MEM_mallocN(sizeof(float[3]) * bl->nr, "dlverts");
|
2006-06-06 10:09:44 +00:00
|
|
|
BLI_addtail(dispbase, dl);
|
2010-03-30 09:57:58 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
if (bl->poly != -1) dl->type = DL_POLY;
|
|
|
|
|
else dl->type = DL_SEGM;
|
2010-03-30 09:57:58 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
if (dl->type == DL_SEGM) dl->flag = (DL_FRONT_CURVE | DL_BACK_CURVE);
|
2010-03-30 09:57:58 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->parts = 1;
|
|
|
|
|
dl->nr = bl->nr;
|
|
|
|
|
dl->col = nu->mat_nr;
|
|
|
|
|
dl->charidx = nu->charidx;
|
2010-03-30 09:57:58 +00:00
|
|
|
|
|
|
|
|
/* dl->rt will be used as flag for render face and */
|
|
|
|
|
/* CU_2D conflicts with R_NOPUNOFLIP */
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->rt = nu->flag & ~CU_2D;
|
2010-03-30 09:57:58 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
a = dl->nr;
|
2014-06-15 14:23:23 +10:00
|
|
|
bevp = bl->bevpoints;
|
2012-05-06 17:22:54 +00:00
|
|
|
data = dl->verts;
|
2012-03-24 06:18:31 +00:00
|
|
|
while (a--) {
|
2012-05-06 17:22:54 +00:00
|
|
|
data[0] = bevp->vec[0] + widfac * bevp->sina;
|
|
|
|
|
data[1] = bevp->vec[1] + widfac * bevp->cosa;
|
|
|
|
|
data[2] = bevp->vec[2];
|
2007-12-03 22:42:48 +00:00
|
|
|
bevp++;
|
2012-05-06 17:22:54 +00:00
|
|
|
data += 3;
|
2007-12-03 22:42:48 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
DispList *dlb;
|
2012-02-02 18:50:44 +00:00
|
|
|
ListBase bottom_capbase = {NULL, NULL};
|
|
|
|
|
ListBase top_capbase = {NULL, NULL};
|
2013-04-26 21:04:12 +00:00
|
|
|
float bottom_no[3] = {0.0f};
|
|
|
|
|
float top_no[3] = {0.0f};
|
2014-03-28 16:41:56 +06:00
|
|
|
float firstblend = 0.0f, lastblend = 0.0f;
|
2015-09-04 01:04:37 +02:00
|
|
|
int i, start, steps = 0;
|
2014-03-28 16:41:56 +06:00
|
|
|
|
2014-05-26 21:50:17 +10:00
|
|
|
if (nu->flagu & CU_NURB_CYCLIC) {
|
|
|
|
|
calc_bevfac_mapping_default(bl,
|
|
|
|
|
&start, &firstblend, &steps, &lastblend);
|
|
|
|
|
}
|
|
|
|
|
else {
|
2014-06-10 22:32:33 +10:00
|
|
|
if (fabsf(cu->bevfac2 - cu->bevfac1) < FLT_EPSILON) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
2014-08-18 17:35:51 +06:00
|
|
|
calc_bevfac_mapping(cu, bl, nu, &start, &firstblend, &steps, &lastblend);
|
2014-05-26 21:50:17 +10:00
|
|
|
}
|
2010-03-30 09:57:58 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
for (dlb = dlbev.first; dlb; dlb = dlb->next) {
|
2014-05-26 21:39:46 +10:00
|
|
|
BevPoint *bevp_first, *bevp_last;
|
|
|
|
|
BevPoint *bevp;
|
2012-05-04 17:04:20 +00:00
|
|
|
|
2010-03-30 09:57:58 +00:00
|
|
|
/* for each part of the bevel use a separate displblock */
|
2012-05-06 17:22:54 +00:00
|
|
|
dl = MEM_callocN(sizeof(DispList), "makeDispListbev1");
|
2014-06-15 14:36:33 +10:00
|
|
|
dl->verts = data = MEM_mallocN(sizeof(float[3]) * dlb->nr * steps, "dlverts");
|
2007-12-03 22:42:48 +00:00
|
|
|
BLI_addtail(dispbase, dl);
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->type = DL_SURF;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->flag = dlb->flag & (DL_FRONT_CURVE | DL_BACK_CURVE);
|
2015-02-27 17:43:10 +11:00
|
|
|
if (dlb->type == DL_POLY) {
|
|
|
|
|
dl->flag |= DL_CYCL_U;
|
|
|
|
|
}
|
2016-06-25 13:57:35 +10:00
|
|
|
if ((bl->poly >= 0) && (steps > 2)) {
|
2015-02-27 17:43:10 +11:00
|
|
|
dl->flag |= DL_CYCL_V;
|
|
|
|
|
}
|
2012-05-04 17:04:20 +00:00
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->parts = steps;
|
|
|
|
|
dl->nr = dlb->nr;
|
|
|
|
|
dl->col = nu->mat_nr;
|
|
|
|
|
dl->charidx = nu->charidx;
|
2010-03-30 09:57:58 +00:00
|
|
|
|
|
|
|
|
/* dl->rt will be used as flag for render face and */
|
|
|
|
|
/* CU_2D conflicts with R_NOPUNOFLIP */
|
2012-05-06 17:22:54 +00:00
|
|
|
dl->rt = nu->flag & ~CU_2D;
|
2010-03-30 09:57:58 +00:00
|
|
|
|
2016-06-25 11:24:25 +10:00
|
|
|
dl->bevel_split = BLI_BITMAP_NEW(steps, "bevel_split");
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2010-03-30 09:57:58 +00:00
|
|
|
/* for each point of poly make a bevel piece */
|
2014-06-15 14:23:23 +10:00
|
|
|
bevp_first = bl->bevpoints;
|
|
|
|
|
bevp_last = &bl->bevpoints[bl->nr - 1];
|
|
|
|
|
bevp = &bl->bevpoints[start];
|
2012-05-06 17:22:54 +00:00
|
|
|
for (i = start, a = 0; a < steps; i++, bevp++, a++) {
|
|
|
|
|
float fac = 1.0;
|
2012-05-04 17:04:20 +00:00
|
|
|
float *cur_data = data;
|
|
|
|
|
|
2012-05-06 17:22:54 +00:00
|
|
|
if (cu->taperobj == NULL) {
|
2013-01-09 10:15:12 +00:00
|
|
|
fac = bevp->radius;
|
2012-03-24 06:18:31 +00:00
|
|
|
}
|
|
|
|
|
else {
|
2012-11-08 08:16:44 +00:00
|
|
|
float len, taper_fac;
|
|
|
|
|
|
|
|
|
|
if (cu->flag & CU_MAP_TAPER) {
|
|
|
|
|
len = (steps - 3) + firstblend + lastblend;
|
|
|
|
|
|
|
|
|
|
if (a == 0)
|
|
|
|
|
taper_fac = 0.0f;
|
|
|
|
|
else if (a == steps - 1)
|
|
|
|
|
taper_fac = 1.0f;
|
|
|
|
|
else
|
|
|
|
|
taper_fac = ((float) a - (1.0f - firstblend)) / len;
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
len = bl->nr - 1;
|
|
|
|
|
taper_fac = (float) i / len;
|
|
|
|
|
|
|
|
|
|
if (a == 0)
|
|
|
|
|
taper_fac += (1.0f - firstblend) / len;
|
|
|
|
|
else if (a == steps - 1)
|
|
|
|
|
taper_fac -= (1.0f - lastblend) / len;
|
|
|
|
|
}
|
|
|
|
|
|
2018-04-06 12:07:27 +02:00
|
|
|
fac = displist_calc_taper(depsgraph, scene, cu->taperobj, taper_fac);
|
2007-12-03 22:42:48 +00:00
|
|
|
}
|
2010-03-30 09:57:58 +00:00
|
|
|
|
2009-09-10 02:57:25 +00:00
|
|
|
if (bevp->split_tag) {
|
2016-06-25 11:24:25 +10:00
|
|
|
BLI_BITMAP_ENABLE(dl->bevel_split, a);
|
2006-06-06 10:09:44 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2012-02-02 15:15:52 +00:00
|
|
|
/* rotate bevel piece and write in data */
|
2014-05-26 21:39:46 +10:00
|
|
|
if ((a == 0) && (bevp != bevp_last)) {
|
2012-11-08 08:16:44 +00:00
|
|
|
rotateBevelPiece(cu, bevp, bevp + 1, dlb, 1.0f - firstblend, widfac, fac, &data);
|
2014-05-26 21:39:46 +10:00
|
|
|
}
|
|
|
|
|
else if ((a == steps - 1) && (bevp != bevp_first) ) {
|
2012-11-08 08:16:44 +00:00
|
|
|
rotateBevelPiece(cu, bevp, bevp - 1, dlb, 1.0f - lastblend, widfac, fac, &data);
|
2014-05-26 21:39:46 +10:00
|
|
|
}
|
|
|
|
|
else {
|
2012-11-08 08:16:44 +00:00
|
|
|
rotateBevelPiece(cu, bevp, NULL, dlb, 0.0f, widfac, fac, &data);
|
2014-05-26 21:39:46 +10:00
|
|
|
}
|
2012-05-04 17:04:20 +00:00
|
|
|
|
2013-04-22 09:26:02 +00:00
|
|
|
if (cu->bevobj && (cu->flag & CU_FILL_CAPS) && !(nu->flagu & CU_NURB_CYCLIC)) {
|
2013-04-26 21:04:12 +00:00
|
|
|
if (a == 1) {
|
2012-05-06 17:22:54 +00:00
|
|
|
fillBevelCap(nu, dlb, cur_data - 3 * dlb->nr, &bottom_capbase);
|
2013-04-26 21:04:12 +00:00
|
|
|
negate_v3_v3(bottom_no, bevp->dir);
|
|
|
|
|
}
|
|
|
|
|
if (a == steps - 1) {
|
2012-05-04 17:04:20 +00:00
|
|
|
fillBevelCap(nu, dlb, cur_data, &top_capbase);
|
2013-04-26 21:04:12 +00:00
|
|
|
copy_v3_v3(top_no, bevp->dir);
|
|
|
|
|
}
|
2006-06-06 10:09:44 +00:00
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-02-02 15:15:52 +00:00
|
|
|
|
2007-12-03 22:42:48 +00:00
|
|
|
/* gl array drawing: using indices */
|
|
|
|
|
displist_surf_indices(dl);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2012-02-02 15:15:52 +00:00
|
|
|
|
2012-03-24 06:18:31 +00:00
|
|
|
if (bottom_capbase.first) {
|
2013-04-26 21:04:12 +00:00
|
|
|
BKE_displist_fill(&bottom_capbase, dispbase, bottom_no, false);
|
|
|
|
|
BKE_displist_fill(&top_capbase, dispbase, top_no, false);
|
2012-05-07 06:58:03 +00:00
|
|
|
BKE_displist_free(&bottom_capbase);
|
|
|
|
|
BKE_displist_free(&top_capbase);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2005-08-14 06:08:41 +00:00
|
|
|
|
2006-06-06 10:09:44 +00:00
|
|
|
}
|
2012-05-07 06:58:03 +00:00
|
|
|
BKE_displist_free(&dlbev);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2005-10-03 14:13:47 +00:00
|
|
|
|
2010-03-16 21:09:53 +00:00
|
|
|
if (!(cu->flag & CU_DEFORM_FILL)) {
|
2013-08-19 09:36:40 +00:00
|
|
|
curve_to_filledpoly(cu, &nubase, dispbase);
|
2010-03-16 21:09:53 +00:00
|
|
|
}
|
2005-10-03 14:13:47 +00:00
|
|
|
|
2014-03-20 22:56:28 +11:00
|
|
|
if (!for_orco) {
|
2014-01-10 01:45:53 +06:00
|
|
|
if ((cu->flag & CU_PATH) ||
|
2018-04-06 12:07:27 +02:00
|
|
|
DEG_get_eval_flags_for_id(depsgraph, &ob->id) & DAG_EVAL_NEED_CURVE_PATH)
|
2014-01-10 01:45:53 +06:00
|
|
|
{
|
|
|
|
|
calc_curvepath(ob, &nubase);
|
|
|
|
|
}
|
|
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
|
2014-10-21 14:44:08 +02:00
|
|
|
if (!for_orco) {
|
2018-07-30 16:54:40 +02:00
|
|
|
BKE_nurbList_duplicate(&ob->runtime.curve_cache->deformed_nurbs, &nubase);
|
2018-10-15 15:58:58 +11:00
|
|
|
curve_calc_modifiers_post(depsgraph, scene, ob, &nubase, dispbase, r_final, for_render, use_render_resolution);
|
2014-10-21 14:44:08 +02:00
|
|
|
}
|
2010-03-16 21:09:53 +00:00
|
|
|
|
2018-10-16 10:33:44 +11:00
|
|
|
if (cu->flag & CU_DEFORM_FILL && !ob->runtime.mesh_eval) {
|
2013-08-19 09:36:40 +00:00
|
|
|
curve_to_filledpoly(cu, &nubase, dispbase);
|
2010-03-16 21:09:53 +00:00
|
|
|
}
|
2013-08-19 09:36:40 +00:00
|
|
|
|
|
|
|
|
BKE_nurbList_free(&nubase);
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
2010-03-08 13:49:13 +00:00
|
|
|
}
|
|
|
|
|
|
2018-11-28 18:14:40 +01:00
|
|
|
void BKE_displist_make_curveTypes(
|
2019-02-14 17:21:55 +11:00
|
|
|
Depsgraph *depsgraph, Scene *scene, Object *ob, const bool for_render, const bool for_orco,
|
|
|
|
|
LinkNode *ob_cyclic_list)
|
2010-03-08 13:49:13 +00:00
|
|
|
{
|
|
|
|
|
ListBase *dispbase;
|
|
|
|
|
|
2011-08-16 10:31:28 +00:00
|
|
|
/* The same check for duplis as in do_makeDispListCurveTypes.
|
2012-03-09 18:28:30 +00:00
|
|
|
* Happens when curve used for constraint/bevel was converted to mesh.
|
|
|
|
|
* check there is still needed for render displist and orco displists. */
|
2014-07-20 01:30:29 +10:00
|
|
|
if (!ELEM(ob->type, OB_SURF, OB_CURVE, OB_FONT))
|
2012-05-07 06:58:03 +00:00
|
|
|
return;
|
2011-08-16 10:31:28 +00:00
|
|
|
|
2013-08-21 07:40:19 +00:00
|
|
|
BKE_object_free_derived_caches(ob);
|
|
|
|
|
|
2018-07-30 16:54:40 +02:00
|
|
|
if (!ob->runtime.curve_cache) {
|
|
|
|
|
ob->runtime.curve_cache = MEM_callocN(sizeof(CurveCache), "CurveCache for curve types");
|
2013-08-19 09:25:24 +00:00
|
|
|
}
|
|
|
|
|
|
2018-07-30 16:54:40 +02:00
|
|
|
dispbase = &(ob->runtime.curve_cache->disp);
|
2010-03-08 13:49:13 +00:00
|
|
|
|
2019-02-14 17:21:55 +11:00
|
|
|
do_makeDispListCurveTypes(
|
|
|
|
|
depsgraph, scene, ob, dispbase, for_render, for_orco, false,
|
|
|
|
|
ob_cyclic_list,
|
|
|
|
|
&ob->runtime.mesh_eval);
|
2010-03-05 16:47:52 +00:00
|
|
|
|
2013-08-19 09:13:15 +00:00
|
|
|
boundbox_displist_object(ob);
|
2010-03-05 16:47:52 +00:00
|
|
|
}
|
|
|
|
|
|
2017-08-16 12:45:11 +10:00
|
|
|
void BKE_displist_make_curveTypes_forRender(
|
2018-04-06 12:07:27 +02:00
|
|
|
Depsgraph *depsgraph, Scene *scene, Object *ob, ListBase *dispbase,
|
2018-10-15 15:58:58 +11:00
|
|
|
Mesh **r_final, const bool for_orco,
|
2019-02-14 17:21:55 +11:00
|
|
|
const bool use_render_resolution,
|
|
|
|
|
LinkNode *ob_cyclic_list)
|
2010-03-08 13:49:13 +00:00
|
|
|
{
|
2018-07-30 16:54:40 +02:00
|
|
|
if (ob->runtime.curve_cache == NULL) {
|
|
|
|
|
ob->runtime.curve_cache = MEM_callocN(sizeof(CurveCache), "CurveCache for Curve");
|
2013-08-19 14:22:02 +00:00
|
|
|
}
|
|
|
|
|
|
2019-02-14 17:21:55 +11:00
|
|
|
do_makeDispListCurveTypes(
|
|
|
|
|
depsgraph, scene, ob, dispbase, true, for_orco, use_render_resolution,
|
|
|
|
|
ob_cyclic_list,
|
|
|
|
|
r_final);
|
2010-03-08 13:49:13 +00:00
|
|
|
}
|
|
|
|
|
|
2017-08-16 12:45:11 +10:00
|
|
|
void BKE_displist_make_curveTypes_forOrco(
|
2019-02-14 17:21:55 +11:00
|
|
|
Depsgraph *depsgraph, Scene *scene, Object *ob, ListBase *dispbase,
|
|
|
|
|
LinkNode *ob_cyclic_list)
|
2010-03-08 13:49:13 +00:00
|
|
|
{
|
2018-07-30 16:54:40 +02:00
|
|
|
if (ob->runtime.curve_cache == NULL) {
|
|
|
|
|
ob->runtime.curve_cache = MEM_callocN(sizeof(CurveCache), "CurveCache for Curve");
|
2013-08-19 14:22:02 +00:00
|
|
|
}
|
|
|
|
|
|
2019-02-14 17:21:55 +11:00
|
|
|
do_makeDispListCurveTypes(
|
|
|
|
|
depsgraph, scene, ob, dispbase, 1, 1, 1,
|
|
|
|
|
ob_cyclic_list,
|
|
|
|
|
NULL);
|
2010-03-08 13:49:13 +00:00
|
|
|
}
|
|
|
|
|
|
2013-08-21 07:40:19 +00:00
|
|
|
void BKE_displist_minmax(ListBase *dispbase, float min[3], float max[3])
|
2002-10-12 11:37:38 +00:00
|
|
|
{
|
|
|
|
|
DispList *dl;
|
2014-04-27 00:20:13 +10:00
|
|
|
const float *vert;
|
2012-05-06 17:22:54 +00:00
|
|
|
int a, tot = 0;
|
2013-01-29 08:21:21 +00:00
|
|
|
int doit = 0;
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2013-01-29 08:21:21 +00:00
|
|
|
for (dl = dispbase->first; dl; dl = dl->next) {
|
|
|
|
|
tot = (dl->type == DL_INDEX3) ? dl->nr : dl->nr * dl->parts;
|
|
|
|
|
vert = dl->verts;
|
|
|
|
|
for (a = 0; a < tot; a++, vert += 3) {
|
|
|
|
|
minmax_v3v3_v3(min, max, vert);
|
|
|
|
|
}
|
|
|
|
|
doit |= (tot != 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!doit) {
|
|
|
|
|
/* there's no geometry in displist, use zero-sized boundbox */
|
|
|
|
|
zero_v3(min);
|
|
|
|
|
zero_v3(max);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* this is confusing, there's also min_max_object, appplying the obmat... */
|
2013-08-19 09:13:15 +00:00
|
|
|
static void boundbox_displist_object(Object *ob)
|
2013-01-29 08:21:21 +00:00
|
|
|
{
|
2014-07-20 01:30:29 +10:00
|
|
|
if (ELEM(ob->type, OB_CURVE, OB_SURF, OB_FONT)) {
|
2014-07-18 17:46:45 +02:00
|
|
|
/* Curve's BB is already calculated as a part of modifier stack,
|
2013-08-19 09:13:15 +00:00
|
|
|
* here we only calculate object BB based on final display list.
|
|
|
|
|
*/
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2013-01-29 08:21:21 +00:00
|
|
|
/* object's BB is calculated from final displist */
|
2019-02-17 12:24:08 +11:00
|
|
|
if (ob->runtime.bb == NULL)
|
|
|
|
|
ob->runtime.bb = MEM_callocN(sizeof(BoundBox), "boundbox");
|
2012-05-07 06:58:03 +00:00
|
|
|
|
2018-10-15 17:14:05 +11:00
|
|
|
if (ob->runtime.mesh_eval) {
|
|
|
|
|
BKE_object_boundbox_calc_from_mesh(ob, ob->runtime.mesh_eval);
|
2013-08-19 09:13:15 +00:00
|
|
|
}
|
|
|
|
|
else {
|
2013-08-21 07:40:19 +00:00
|
|
|
float min[3], max[3];
|
|
|
|
|
|
|
|
|
|
INIT_MINMAX(min, max);
|
2018-07-30 16:54:40 +02:00
|
|
|
BKE_displist_minmax(&ob->runtime.curve_cache->disp, min, max);
|
2019-02-17 12:24:08 +11:00
|
|
|
BKE_boundbox_init_from_minmax(ob->runtime.bb, min, max);
|
2016-03-04 21:50:54 +11:00
|
|
|
|
2019-02-17 12:24:08 +11:00
|
|
|
ob->runtime.bb->flag &= ~BOUNDBOX_DIRTY;
|
2013-08-19 09:13:15 +00:00
|
|
|
}
|
2002-10-12 11:37:38 +00:00
|
|
|
}
|
|
|
|
|
}
|