2 * ***** BEGIN GPL LICENSE BLOCK *****
4 * This program is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU General Public License
6 * as published by the Free Software Foundation; either version 2
7 * of the License, or (at your option) any later version.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, write to the Free Software Foundation,
16 * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18 * The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
19 * All rights reserved.
21 * The Original Code is: all of this file.
23 * Contributor(s): none yet.
25 * ***** END GPL LICENSE BLOCK *****
28 /** \file blender/blenkernel/intern/displist.c
37 #include "MEM_guardedalloc.h"
39 #include "DNA_curve_types.h"
40 #include "DNA_scene_types.h"
41 #include "DNA_object_types.h"
42 #include "DNA_vfont_types.h"
44 #include "BLI_blenlib.h"
45 #include "BLI_memarena.h"
47 #include "BLI_scanfill.h"
48 #include "BLI_utildefines.h"
50 #include "BKE_global.h"
51 #include "BKE_depsgraph.h"
52 #include "BKE_displist.h"
53 #include "BKE_cdderivedmesh.h"
54 #include "BKE_object.h"
55 #include "BKE_mball.h"
56 #include "BKE_mball_tessellate.h"
57 #include "BKE_curve.h"
61 #include "BKE_lattice.h"
62 #include "BKE_modifier.h"
64 #include "BLI_sys_types.h" // for intptr_t support
66 static void boundbox_displist_object(Object *ob);
68 void BKE_displist_elem_free(DispList *dl)
71 if (dl->verts) MEM_freeN(dl->verts);
72 if (dl->nors) MEM_freeN(dl->nors);
73 if (dl->index) MEM_freeN(dl->index);
74 if (dl->bevelSplitFlag) MEM_freeN(dl->bevelSplitFlag);
79 void BKE_displist_free(ListBase *lb)
83 while ((dl = BLI_pophead(lb))) {
84 BKE_displist_elem_free(dl);
88 DispList *BKE_displist_find_or_create(ListBase *lb, int type)
99 dl = MEM_callocN(sizeof(DispList), "find_disp");
106 DispList *BKE_displist_find(ListBase *lb, int type)
112 if (dl->type == type)
120 bool BKE_displist_has_faces(ListBase *lb)
124 for (dl = lb->first; dl; dl = dl->next) {
125 if (ELEM(dl->type, DL_INDEX3, DL_INDEX4, DL_SURF)) {
133 void BKE_displist_copy(ListBase *lbn, ListBase *lb)
137 BKE_displist_free(lbn);
141 dln = MEM_dupallocN(dl);
142 BLI_addtail(lbn, dln);
143 dln->verts = MEM_dupallocN(dl->verts);
144 dln->nors = MEM_dupallocN(dl->nors);
145 dln->index = MEM_dupallocN(dl->index);
147 if (dl->bevelSplitFlag)
148 dln->bevelSplitFlag = MEM_dupallocN(dl->bevelSplitFlag);
154 void BKE_displist_normals_add(ListBase *lb)
157 float *vdata, *ndata, nor[3];
158 float *v1, *v2, *v3, *v4;
159 float *n1, *n2, *n3, *n4;
160 int a, b, p1, p2, p3, p4;
165 if (dl->type == DL_INDEX3) {
166 if (dl->nors == NULL) {
167 dl->nors = MEM_callocN(sizeof(float) * 3, "dlnors");
169 if (dl->verts[2] < 0.0f)
175 else if (dl->type == DL_SURF) {
176 if (dl->nors == NULL) {
177 dl->nors = MEM_callocN(sizeof(float) * 3 * dl->nr * dl->parts, "dlnors");
182 for (a = 0; a < dl->parts; a++) {
184 if (BKE_displist_surfindex_get(dl, a, &b, &p1, &p2, &p3, &p4) == 0)
196 for (; b < dl->nr; b++) {
197 normal_quad_v3(nor, v1, v3, v4, v2);
210 a = dl->parts * dl->nr;
222 void BKE_displist_count(ListBase *lb, int *totvert, int *totface, int *tottri)
226 for (dl = lb->first; dl; dl = dl->next) {
234 vert_tot = dl->nr * dl->parts;
235 face_tot = (dl->nr - 1) * (dl->parts - 1);
236 tri_tot = face_tot * 2;
242 face_tot = dl->parts;
249 face_tot = dl->parts;
250 tri_tot = face_tot * 2;
256 vert_tot = dl->nr * dl->parts;
261 *totvert += vert_tot;
262 *totface += face_tot;
267 bool BKE_displist_surfindex_get(DispList *dl, int a, int *b, int *p1, int *p2, int *p3, int *p4)
269 if ((dl->flag & DL_CYCL_V) == 0 && a == (dl->parts) - 1) {
273 if (dl->flag & DL_CYCL_U) {
275 (*p2) = (*p1) + dl->nr - 1;
276 (*p3) = (*p1) + dl->nr;
277 (*p4) = (*p2) + dl->nr;
283 (*p4) = (*p2) + dl->nr;
284 (*p3) = (*p1) + dl->nr;
288 if ((dl->flag & DL_CYCL_V) && a == dl->parts - 1) {
289 (*p3) -= dl->nr * dl->parts;
290 (*p4) -= dl->nr * dl->parts;
296 /* ****************** make displists ********************* */
298 static void curve_to_displist(Curve *cu, ListBase *nubase, ListBase *dispbase,
299 const bool for_render, const bool use_render_resolution)
303 BezTriple *bezt, *prevbezt;
307 const bool editmode = (!for_render && (cu->editnurb || cu->editfont));
311 if (nu->hide == 0 || editmode == false) {
312 if (use_render_resolution && cu->resolu_ren != 0)
313 resolu = cu->resolu_ren;
317 if (!BKE_nurb_check_valid_u(nu)) {
320 else if (nu->type == CU_BEZIER) {
324 if (nu->flagu & CU_NURB_CYCLIC) a++;
329 if (a == 0 && (nu->flagu & CU_NURB_CYCLIC))
332 if (prevbezt->h2 == HD_VECT && bezt->h1 == HD_VECT)
337 if (a == 0 && (nu->flagu & CU_NURB_CYCLIC) == 0)
344 dl = MEM_callocN(sizeof(DispList), "makeDispListbez");
345 /* len+1 because of 'forward_diff_bezier' function */
346 dl->verts = MEM_mallocN((len + 1) * sizeof(float[3]), "dlverts");
347 BLI_addtail(dispbase, dl);
350 dl->col = nu->mat_nr;
351 dl->charidx = nu->charidx;
355 /* check that (len != 2) so we don't immediately loop back on ourselves */
356 if (nu->flagu & CU_NURB_CYCLIC && (dl->nr != 2)) {
369 if (a == 0 && dl->type == DL_POLY)
372 if (prevbezt->h2 == HD_VECT && bezt->h1 == HD_VECT) {
373 copy_v3_v3(data, prevbezt->vec[1]);
378 for (j = 0; j < 3; j++) {
379 BKE_curve_forward_diff_bezier(prevbezt->vec[1][j],
383 data + j, resolu, 3 * sizeof(float));
389 if (a == 0 && dl->type == DL_SEGM) {
390 copy_v3_v3(data, bezt->vec[1]);
397 else if (nu->type == CU_NURBS) {
398 len = (resolu * SEGMENTSU(nu));
400 dl = MEM_callocN(sizeof(DispList), "makeDispListsurf");
401 dl->verts = MEM_mallocN(len * sizeof(float[3]), "dlverts");
402 BLI_addtail(dispbase, dl);
406 dl->col = nu->mat_nr;
407 dl->charidx = nu->charidx;
410 if (nu->flagu & CU_NURB_CYCLIC)
412 else dl->type = DL_SEGM;
413 BKE_nurb_makeCurve(nu, data, NULL, NULL, NULL, resolu, 3 * sizeof(float));
415 else if (nu->type == CU_POLY) {
417 dl = MEM_callocN(sizeof(DispList), "makeDispListpoly");
418 dl->verts = MEM_mallocN(len * sizeof(float[3]), "dlverts");
419 BLI_addtail(dispbase, dl);
422 dl->col = nu->mat_nr;
423 dl->charidx = nu->charidx;
426 if ((nu->flagu & CU_NURB_CYCLIC) && (dl->nr != 2)) {
436 copy_v3_v3(data, bp->vec);
447 * \param normal_proj Optional normal thats used to project the scanfill verts into 2d coords.
448 * Pass this along if known since it saves time calculating the normal.
449 * \param flipnormal Flip the normal (same as passing \a normal_proj negated)
451 void BKE_displist_fill(ListBase *dispbase, ListBase *to, const float normal_proj[3], const bool flipnormal)
453 ScanFillContext sf_ctx;
454 ScanFillVert *sf_vert, *sf_vert_new, *sf_vert_last;
455 ScanFillFace *sf_tri;
457 DispList *dlnew = NULL, *dl;
459 int colnr = 0, charidx = 0, cont = 1, tot, a, *index, nextcol = 0;
461 const int scanfill_flag = BLI_SCANFILL_CALC_REMOVE_DOUBLES | BLI_SCANFILL_CALC_POLYS | BLI_SCANFILL_CALC_HOLES;
463 if (dispbase == NULL)
465 if (BLI_listbase_is_empty(dispbase))
468 sf_arena = BLI_memarena_new(BLI_SCANFILL_ARENA_SIZE, __func__);
475 BLI_scanfill_begin_arena(&sf_ctx, sf_arena);
477 dl = dispbase->first;
479 if (dl->type == DL_POLY) {
480 if (charidx < dl->charidx)
482 else if (charidx == dl->charidx) { /* character with needed index */
483 if (colnr == dl->col) {
487 /* make editverts and edges */
490 sf_vert = sf_vert_new = NULL;
493 sf_vert_last = sf_vert;
495 sf_vert = BLI_scanfill_vert_add(&sf_ctx, f1);
498 if (sf_vert_last == NULL)
499 sf_vert_new = sf_vert;
501 BLI_scanfill_edge_add(&sf_ctx, sf_vert_last, sf_vert);
506 if (sf_vert != NULL && sf_vert_new != NULL) {
507 BLI_scanfill_edge_add(&sf_ctx, sf_vert, sf_vert_new);
510 else if (colnr < dl->col) {
511 /* got poly with next material at current char */
520 /* XXX (obedit && obedit->actcol) ? (obedit->actcol-1) : 0)) { */
521 if (totvert && (tot = BLI_scanfill_calc_ex(&sf_ctx,
526 dlnew = MEM_callocN(sizeof(DispList), "filldisplist");
527 dlnew->type = DL_INDEX3;
532 dlnew->index = MEM_mallocN(tot * 3 * sizeof(int), "dlindex");
533 dlnew->verts = MEM_mallocN(totvert * 3 * sizeof(float), "dlverts");
539 for (sf_vert = sf_ctx.fillvertbase.first; sf_vert; sf_vert = sf_vert->next) {
540 copy_v3_v3(f1, sf_vert->co);
544 sf_vert->tmp.i = totvert;
550 index = dlnew->index;
551 for (sf_tri = sf_ctx.fillfacebase.first; sf_tri; sf_tri = sf_tri->next) {
552 index[0] = sf_tri->v1->tmp.i;
553 index[1] = sf_tri->v2->tmp.i;
554 index[2] = sf_tri->v3->tmp.i;
557 SWAP(int, index[0], index[2]);
563 BLI_addhead(to, dlnew);
565 BLI_scanfill_end_arena(&sf_ctx, sf_arena);
568 /* stay at current char but fill polys with next material */
572 /* switch to next char and start filling from first material */
578 BLI_memarena_free(sf_arena);
580 /* do not free polys, needed for wireframe display */
583 static void bevels_to_filledpoly(Curve *cu, ListBase *dispbase)
585 const float z_up[3] = {0.0f, 0.0f, 1.0f};
586 ListBase front, back;
587 DispList *dl, *dlnew;
591 BLI_listbase_clear(&front);
592 BLI_listbase_clear(&back);
594 dl = dispbase->first;
596 if (dl->type == DL_SURF) {
597 if ((dl->flag & DL_CYCL_V) && (dl->flag & DL_CYCL_U) == 0) {
598 if ((cu->flag & CU_BACK) && (dl->flag & DL_BACK_CURVE)) {
599 dlnew = MEM_callocN(sizeof(DispList), "filldisp");
600 BLI_addtail(&front, dlnew);
601 dlnew->verts = fp1 = MEM_mallocN(sizeof(float) * 3 * dl->parts, "filldisp1");
602 dlnew->nr = dl->parts;
604 dlnew->type = DL_POLY;
605 dlnew->col = dl->col;
606 dlnew->charidx = dl->charidx;
618 if ((cu->flag & CU_FRONT) && (dl->flag & DL_FRONT_CURVE)) {
619 dlnew = MEM_callocN(sizeof(DispList), "filldisp");
620 BLI_addtail(&back, dlnew);
621 dlnew->verts = fp1 = MEM_mallocN(sizeof(float) * 3 * dl->parts, "filldisp1");
622 dlnew->nr = dl->parts;
624 dlnew->type = DL_POLY;
625 dlnew->col = dl->col;
626 dlnew->charidx = dl->charidx;
628 fp = dl->verts + 3 * (dl->nr - 1);
643 BKE_displist_fill(&front, dispbase, z_up, true);
644 BKE_displist_fill(&back, dispbase, z_up, false);
646 BKE_displist_free(&front);
647 BKE_displist_free(&back);
649 BKE_displist_fill(dispbase, dispbase, z_up, false);
652 static void curve_to_filledpoly(Curve *cu, ListBase *UNUSED(nurb), ListBase *dispbase)
654 if (!CU_DO_2DFILL(cu))
657 if (dispbase->first && ((DispList *) dispbase->first)->type == DL_SURF) {
658 bevels_to_filledpoly(cu, dispbase);
661 const float z_up[3] = {0.0f, 0.0f, 1.0f};
662 BKE_displist_fill(dispbase, dispbase, z_up, false);
668 * - first point left, last point right
669 * - based on subdivided points in original curve, not on points in taper curve (still)
671 static float displist_calc_taper(Scene *scene, Object *taperobj, float fac)
675 if (taperobj == NULL || taperobj->type != OB_CURVE)
678 dl = taperobj->curve_cache ? taperobj->curve_cache->disp.first : NULL;
680 BKE_displist_make_curveTypes(scene, taperobj, 0);
681 dl = taperobj->curve_cache->disp.first;
687 /* horizontal size */
689 dx = dl->verts[3 * (dl->nr - 1)] - minx;
692 for (a = 0; a < dl->nr; a++, fp += 3) {
693 if ((fp[0] - minx) / dx >= fac) {
694 /* interpolate with prev */
696 float fac1 = (fp[-3] - minx) / dx;
697 float fac2 = (fp[0] - minx) / dx;
699 return fp[1] * (fac1 - fac) / (fac1 - fac2) + fp[-2] * (fac - fac2) / (fac1 - fac2);
704 return fp[-2]; // last y coord
711 float BKE_displist_calc_taper(Scene *scene, Object *taperobj, int cur, int tot)
713 float fac = ((float)cur) / (float)(tot - 1);
715 return displist_calc_taper(scene, taperobj, fac);
718 void BKE_displist_make_mball(EvaluationContext *eval_ctx, Scene *scene, Object *ob)
720 if (!ob || ob->type != OB_MBALL)
723 if (ob == BKE_mball_basis_find(scene, ob)) {
724 if (ob->curve_cache) {
725 BKE_displist_free(&(ob->curve_cache->disp));
728 ob->curve_cache = MEM_callocN(sizeof(CurveCache), "CurveCache for MBall");
731 BKE_mball_polygonize(eval_ctx, scene, ob, &ob->curve_cache->disp);
732 BKE_mball_texspace_calc(ob);
734 object_deform_mball(ob, &ob->curve_cache->disp);
736 /* NOP for MBALLs anyway... */
737 boundbox_displist_object(ob);
741 void BKE_displist_make_mball_forRender(EvaluationContext *eval_ctx, Scene *scene, Object *ob, ListBase *dispbase)
743 BKE_mball_polygonize(eval_ctx, scene, ob, dispbase);
744 BKE_mball_texspace_calc(ob);
746 object_deform_mball(ob, dispbase);
749 static ModifierData *curve_get_tessellate_point(Scene *scene, Object *ob,
750 const bool use_render_resolution, const bool editmode)
752 VirtualModifierData virtualModifierData;
753 ModifierData *md = modifiers_getVirtualModifierList(ob, &virtualModifierData);
754 ModifierData *pretessellatePoint;
757 if (use_render_resolution)
758 required_mode = eModifierMode_Render;
760 required_mode = eModifierMode_Realtime;
763 required_mode |= eModifierMode_Editmode;
765 pretessellatePoint = NULL;
766 for (; md; md = md->next) {
767 ModifierTypeInfo *mti = modifierType_getInfo(md->type);
769 if (!modifier_isEnabled(scene, md, required_mode))
771 if (mti->type == eModifierTypeType_Constructive)
772 return pretessellatePoint;
774 if (ELEM(md->type, eModifierType_Hook, eModifierType_Softbody, eModifierType_MeshDeform)) {
775 pretessellatePoint = md;
777 /* this modifiers are moving point of tessellation automatically
778 * (some of them even can't be applied on tessellated curve), set flag
779 * for information button in modifier's header
781 md->mode |= eModifierMode_ApplyOnSpline;
783 else if (md->mode & eModifierMode_ApplyOnSpline) {
784 pretessellatePoint = md;
788 return pretessellatePoint;
791 static void curve_calc_modifiers_pre(Scene *scene, Object *ob, ListBase *nurb,
792 const bool for_render, const bool use_render_resolution)
794 VirtualModifierData virtualModifierData;
795 ModifierData *md = modifiers_getVirtualModifierList(ob, &virtualModifierData);
796 ModifierData *pretessellatePoint;
797 Curve *cu = ob->data;
799 const bool editmode = (!for_render && (cu->editnurb || cu->editfont));
800 ModifierApplyFlag app_flag = 0;
801 float (*deformedVerts)[3] = NULL;
802 float *keyVerts = NULL;
805 modifiers_clearErrors(ob);
808 app_flag |= MOD_APPLY_USECACHE;
809 if (use_render_resolution) {
810 app_flag |= MOD_APPLY_RENDER;
811 required_mode = eModifierMode_Render;
814 required_mode = eModifierMode_Realtime;
816 pretessellatePoint = curve_get_tessellate_point(scene, ob, use_render_resolution, editmode);
819 required_mode |= eModifierMode_Editmode;
821 if (cu->editnurb == NULL) {
822 keyVerts = BKE_key_evaluate_object(ob, &numVerts);
825 /* split coords from key data, the latter also includes
826 * tilts, which is passed through in the modifier stack.
827 * this is also the reason curves do not use a virtual
828 * shape key modifier yet. */
829 deformedVerts = BKE_curve_nurbs_keyVertexCos_get(nurb, keyVerts);
830 BLI_assert(BKE_nurbList_verts_count(nurb) == numVerts);
834 if (pretessellatePoint) {
835 for (; md; md = md->next) {
836 ModifierTypeInfo *mti = modifierType_getInfo(md->type);
840 if (!modifier_isEnabled(scene, md, required_mode))
842 if (mti->type != eModifierTypeType_OnlyDeform)
845 if (!deformedVerts) {
846 deformedVerts = BKE_curve_nurbs_vertexCos_get(nurb, &numVerts);
849 mti->deformVerts(md, ob, NULL, deformedVerts, numVerts, app_flag);
851 if (md == pretessellatePoint)
857 BK_curve_nurbs_vertexCos_apply(nurb, deformedVerts);
858 MEM_freeN(deformedVerts);
860 if (keyVerts) /* these are not passed through modifier stack */
861 BKE_curve_nurbs_keyVertexTilts_apply(nurb, keyVerts);
867 static float (*displist_get_allverts(ListBase *dispbase, int *totvert))[3]
870 float (*allverts)[3], *fp;
874 for (dl = dispbase->first; dl; dl = dl->next)
875 *totvert += (dl->type == DL_INDEX3) ? dl->nr : dl->parts * dl->nr;
877 allverts = MEM_mallocN((*totvert) * sizeof(float) * 3, "displist_get_allverts allverts");
878 fp = (float *)allverts;
879 for (dl = dispbase->first; dl; dl = dl->next) {
880 int offs = 3 * ((dl->type == DL_INDEX3) ? dl->nr : dl->parts * dl->nr);
881 memcpy(fp, dl->verts, sizeof(float) * offs);
888 static void displist_apply_allverts(ListBase *dispbase, float (*allverts)[3])
893 fp = (float *)allverts;
894 for (dl = dispbase->first; dl; dl = dl->next) {
895 int offs = 3 * ((dl->type == DL_INDEX3) ? dl->nr : dl->parts * dl->nr);
896 memcpy(dl->verts, fp, sizeof(float) * offs);
901 static void curve_calc_modifiers_post(Scene *scene, Object *ob, ListBase *nurb,
902 ListBase *dispbase, DerivedMesh **r_dm_final,
903 const bool for_render, const bool use_render_resolution)
905 VirtualModifierData virtualModifierData;
906 ModifierData *md = modifiers_getVirtualModifierList(ob, &virtualModifierData);
907 ModifierData *pretessellatePoint;
908 Curve *cu = ob->data;
909 int required_mode = 0, totvert = 0;
910 const bool editmode = (!for_render && (cu->editnurb || cu->editfont));
911 DerivedMesh *dm = NULL, *ndm;
912 float (*vertCos)[3] = NULL;
913 int useCache = !for_render;
914 ModifierApplyFlag app_flag = 0;
916 if (use_render_resolution) {
917 app_flag |= MOD_APPLY_RENDER;
918 required_mode = eModifierMode_Render;
921 required_mode = eModifierMode_Realtime;
923 pretessellatePoint = curve_get_tessellate_point(scene, ob, use_render_resolution, editmode);
926 required_mode |= eModifierMode_Editmode;
928 if (pretessellatePoint) {
929 md = pretessellatePoint->next;
932 if (r_dm_final && *r_dm_final) {
933 (*r_dm_final)->release(*r_dm_final);
936 for (; md; md = md->next) {
937 ModifierTypeInfo *mti = modifierType_getInfo(md->type);
938 ModifierApplyFlag appf = app_flag;
942 if (!modifier_isEnabled(scene, md, required_mode))
945 if (mti->type == eModifierTypeType_OnlyDeform ||
946 (mti->type == eModifierTypeType_DeformOrConstruct && !dm))
949 appf |= MOD_APPLY_USECACHE;
952 totvert = dm->getNumVerts(dm);
953 vertCos = MEM_mallocN(sizeof(*vertCos) * totvert, "dfmv");
954 dm->getVertCos(dm, vertCos);
957 mti->deformVerts(md, ob, dm, vertCos, totvert, appf);
961 vertCos = displist_get_allverts(dispbase, &totvert);
964 mti->deformVerts(md, ob, NULL, vertCos, totvert, appf);
969 /* makeDisplistCurveTypes could be used for beveling, where derived mesh
970 * is totally unnecessary, so we could stop modifiers applying
971 * when we found constructive modifier but derived mesh is unwanted result
978 DerivedMesh *tdm = CDDM_copy(dm);
982 CDDM_apply_vert_coords(dm, vertCos);
987 displist_apply_allverts(dispbase, vertCos);
990 if (ELEM(ob->type, OB_CURVE, OB_FONT) && (cu->flag & CU_DEFORM_FILL)) {
991 curve_to_filledpoly(cu, nurb, dispbase);
994 dm = CDDM_from_curve_displist(ob, dispbase);
998 /* Vertex coordinates were applied to necessary data, could free it */
1004 appf |= MOD_APPLY_USECACHE;
1006 ndm = modwrap_applyModifier(md, ob, dm, appf);
1009 /* Modifier returned a new derived mesh */
1011 if (dm && dm != ndm) /* Modifier */
1020 DerivedMesh *tdm = CDDM_copy(dm);
1024 CDDM_apply_vert_coords(dm, vertCos);
1025 CDDM_calc_normals_mapping(dm);
1029 displist_apply_allverts(dispbase, vertCos);
1037 /* see: mesh_calc_modifiers */
1038 if (dm->getNumTessFaces(dm) == 0) {
1039 dm->recalcTessellation(dm);
1041 /* Even if tessellation is not needed, some modifiers might have modified CD layers
1042 * (like mloopcol or mloopuv), hence we have to update those. */
1043 else if (dm->dirty & DM_DIRTY_TESS_CDLAYERS) {
1044 DM_update_tessface_data(dm);
1047 if (dm->type == DM_TYPE_CDDM) {
1048 CDDM_calc_normals_mapping_ex(dm, (dm->dirty & DM_DIRTY_NORMALS) ? false : true);
1055 static void displist_surf_indices(DispList *dl)
1057 int a, b, p1, p2, p3, p4;
1062 index = dl->index = MEM_mallocN(4 * sizeof(int) * (dl->parts + 1) * (dl->nr + 1), "index array nurbs");
1064 for (a = 0; a < dl->parts; a++) {
1066 if (BKE_displist_surfindex_get(dl, a, &b, &p1, &p2, &p3, &p4) == 0)
1069 for (; b < dl->nr; b++, index += 4) {
1083 static DerivedMesh *create_orco_dm(Scene *scene, Object *ob)
1086 ListBase disp = {NULL, NULL};
1088 /* OrcoDM should be created from underformed disp lists */
1089 BKE_displist_make_curveTypes_forOrco(scene, ob, &disp);
1090 dm = CDDM_from_curve_displist(ob, &disp);
1092 BKE_displist_free(&disp);
1097 static void add_orco_dm(Object *ob, DerivedMesh *dm, DerivedMesh *orcodm)
1099 float (*orco)[3], (*layerorco)[3];
1101 Curve *cu = ob->data;
1103 totvert = dm->getNumVerts(dm);
1105 orco = MEM_callocN(sizeof(float) * 3 * totvert, "dm orco");
1107 if (orcodm->getNumVerts(orcodm) == totvert)
1108 orcodm->getVertCos(orcodm, orco);
1110 dm->getVertCos(dm, orco);
1112 for (a = 0; a < totvert; a++) {
1113 float *co = orco[a];
1114 co[0] = (co[0] - cu->loc[0]) / cu->size[0];
1115 co[1] = (co[1] - cu->loc[1]) / cu->size[1];
1116 co[2] = (co[2] - cu->loc[2]) / cu->size[2];
1119 if ((layerorco = DM_get_vert_data_layer(dm, CD_ORCO))) {
1120 memcpy(layerorco, orco, sizeof(float) * totvert);
1124 DM_add_vert_layer(dm, CD_ORCO, CD_ASSIGN, orco);
1127 static void curve_calc_orcodm(Scene *scene, Object *ob, DerivedMesh *dm_final,
1128 const bool for_render, const bool use_render_resolution)
1130 /* this function represents logic of mesh's orcodm calculation
1131 * for displist-based objects
1133 VirtualModifierData virtualModifierData;
1134 ModifierData *md = modifiers_getVirtualModifierList(ob, &virtualModifierData);
1135 ModifierData *pretessellatePoint;
1136 Curve *cu = ob->data;
1138 const bool editmode = (!for_render && (cu->editnurb || cu->editfont));
1139 DerivedMesh *ndm, *orcodm = NULL;
1140 ModifierApplyFlag app_flag = MOD_APPLY_ORCO;
1142 if (use_render_resolution) {
1143 app_flag |= MOD_APPLY_RENDER;
1144 required_mode = eModifierMode_Render;
1147 required_mode = eModifierMode_Realtime;
1149 pretessellatePoint = curve_get_tessellate_point(scene, ob, use_render_resolution, editmode);
1152 required_mode |= eModifierMode_Editmode;
1154 if (pretessellatePoint) {
1155 md = pretessellatePoint->next;
1158 /* If modifiers are disabled, we wouldn't be here because
1159 * this function is only called if there're enabled constructive
1160 * modifiers applied on the curve.
1162 * This means we can create ORCO DM in advance and assume it's
1165 orcodm = create_orco_dm(scene, ob);
1167 for (; md; md = md->next) {
1168 ModifierTypeInfo *mti = modifierType_getInfo(md->type);
1172 if (!modifier_isEnabled(scene, md, required_mode))
1174 if (mti->type != eModifierTypeType_Constructive)
1177 ndm = modwrap_applyModifier(md, ob, orcodm, app_flag);
1180 /* if the modifier returned a new dm, release the old one */
1181 if (orcodm && orcodm != ndm) {
1182 orcodm->release(orcodm);
1188 /* add an orco layer if needed */
1189 add_orco_dm(ob, dm_final, orcodm);
1191 orcodm->release(orcodm);
1194 void BKE_displist_make_surf(Scene *scene, Object *ob, ListBase *dispbase,
1195 DerivedMesh **r_dm_final,
1196 const bool for_render, const bool for_orco, const bool use_render_resolution)
1198 ListBase nubase = {NULL, NULL};
1200 Curve *cu = ob->data;
1205 if (!for_render && cu->editnurb) {
1206 BKE_nurbList_duplicate(&nubase, BKE_curve_editNurbs_get(cu));
1209 BKE_nurbList_duplicate(&nubase, &cu->nurb);
1213 curve_calc_modifiers_pre(scene, ob, &nubase, for_render, use_render_resolution);
1215 for (nu = nubase.first; nu; nu = nu->next) {
1216 if ((for_render || nu->hide == 0) && BKE_nurb_check_valid_uv(nu)) {
1217 int resolu = nu->resolu, resolv = nu->resolv;
1219 if (use_render_resolution) {
1221 resolu = cu->resolu_ren;
1223 resolv = cu->resolv_ren;
1226 if (nu->pntsv == 1) {
1227 len = SEGMENTSU(nu) * resolu;
1229 dl = MEM_callocN(sizeof(DispList), "makeDispListsurf");
1230 dl->verts = MEM_mallocN(len * sizeof(float[3]), "dlverts");
1232 BLI_addtail(dispbase, dl);
1235 dl->col = nu->mat_nr;
1236 dl->charidx = nu->charidx;
1238 /* dl->rt will be used as flag for render face and */
1239 /* CU_2D conflicts with R_NOPUNOFLIP */
1240 dl->rt = nu->flag & ~CU_2D;
1243 if (nu->flagu & CU_NURB_CYCLIC) dl->type = DL_POLY;
1244 else dl->type = DL_SEGM;
1246 BKE_nurb_makeCurve(nu, data, NULL, NULL, NULL, resolu, 3 * sizeof(float));
1249 len = (nu->pntsu * resolu) * (nu->pntsv * resolv);
1251 dl = MEM_callocN(sizeof(DispList), "makeDispListsurf");
1252 dl->verts = MEM_mallocN(len * sizeof(float[3]), "dlverts");
1253 BLI_addtail(dispbase, dl);
1255 dl->col = nu->mat_nr;
1256 dl->charidx = nu->charidx;
1258 /* dl->rt will be used as flag for render face and */
1259 /* CU_2D conflicts with R_NOPUNOFLIP */
1260 dl->rt = nu->flag & ~CU_2D;
1265 dl->parts = (nu->pntsu * resolu); /* in reverse, because makeNurbfaces works that way */
1266 dl->nr = (nu->pntsv * resolv);
1267 if (nu->flagv & CU_NURB_CYCLIC) dl->flag |= DL_CYCL_U; /* reverse too! */
1268 if (nu->flagu & CU_NURB_CYCLIC) dl->flag |= DL_CYCL_V;
1270 BKE_nurb_makeFaces(nu, data, 0, resolu, resolv);
1272 /* gl array drawing: using indices */
1273 displist_surf_indices(dl);
1279 BKE_nurbList_duplicate(&ob->curve_cache->deformed_nurbs, &nubase);
1280 curve_calc_modifiers_post(scene, ob, &nubase, dispbase, r_dm_final,
1281 for_render, use_render_resolution);
1284 BKE_nurbList_free(&nubase);
1287 static void rotateBevelPiece(Curve *cu, BevPoint *bevp, BevPoint *nbevp, DispList *dlb, float bev_blend, float widfac, float fac, float **r_data)
1289 float *fp, *data = *r_data;
1293 for (b = 0; b < dlb->nr; b++, fp += 3, data += 3) {
1294 if (cu->flag & CU_3D) {
1295 float vec[3], quat[4];
1297 vec[0] = fp[1] + widfac;
1301 if (nbevp == NULL) {
1302 copy_v3_v3(data, bevp->vec);
1303 copy_qt_qt(quat, bevp->quat);
1306 interp_v3_v3v3(data, bevp->vec, nbevp->vec, bev_blend);
1307 interp_qt_qtqt(quat, bevp->quat, nbevp->quat, bev_blend);
1310 mul_qt_v3(quat, vec);
1312 data[0] += fac * vec[0];
1313 data[1] += fac * vec[1];
1314 data[2] += fac * vec[2];
1319 if (nbevp == NULL) {
1320 copy_v3_v3(data, bevp->vec);
1325 interp_v3_v3v3(data, bevp->vec, nbevp->vec, bev_blend);
1327 /* perhaps we need to interpolate angles instead. but the thing is
1328 * cosa and sina are not actually sine and cosine
1330 sina = nbevp->sina * bev_blend + bevp->sina * (1.0f - bev_blend);
1331 cosa = nbevp->cosa * bev_blend + bevp->cosa * (1.0f - bev_blend);
1334 data[0] += fac * (widfac + fp[1]) * sina;
1335 data[1] += fac * (widfac + fp[1]) * cosa;
1336 data[2] += fac * fp[2];
1343 static void fillBevelCap(Nurb *nu, DispList *dlb, float *prev_fp, ListBase *dispbase)
1347 dl = MEM_callocN(sizeof(DispList), "makeDispListbev2");
1348 dl->verts = MEM_mallocN(sizeof(float[3]) * dlb->nr, "dlverts");
1349 memcpy(dl->verts, prev_fp, 3 * sizeof(float) * dlb->nr);
1355 dl->col = nu->mat_nr;
1356 dl->charidx = nu->charidx;
1358 /* dl->rt will be used as flag for render face and */
1359 /* CU_2D conflicts with R_NOPUNOFLIP */
1360 dl->rt = nu->flag & ~CU_2D;
1362 BLI_addtail(dispbase, dl);
1365 static void calc_bevfac_segment_mapping(BevList *bl, float bevfac, float spline_length, int *r_bev, float *r_blend)
1367 float normlen, normsum = 0.0f;
1368 float *seglen = bl->seglen;
1369 int *segbevcount = bl->segbevcount;
1370 int bevcount = 0, nr = bl->nr;
1372 float bev_fl = bevfac * (bl->nr - 1);
1373 *r_bev = (int)bev_fl;
1375 while (bevcount < nr - 1) {
1376 normlen = *seglen / spline_length;
1377 if (normsum + normlen > bevfac) {
1378 bev_fl = bevcount + (bevfac - normsum) / normlen * *segbevcount;
1379 *r_bev = (int) bev_fl;
1380 *r_blend = bev_fl - *r_bev;
1384 bevcount += *segbevcount;
1390 static void calc_bevfac_spline_mapping(BevList *bl, float bevfac,
1391 float spline_length,
1392 int *r_bev, float *r_blend)
1394 const float len_target = bevfac * spline_length;
1395 BevPoint *bevp = bl->bevpoints;
1396 float len_next = 0.0f, len = 0.0f;
1397 int i = 0, nr = bl->nr;
1401 len_next = len + bevp->offset;
1402 if (len_next > len_target) {
1410 *r_blend = (len_target - len) / bevp->offset;
1413 static void calc_bevfac_mapping_default(BevList *bl,
1414 int *r_start, float *r_firstblend,
1415 int *r_steps, float *r_lastblend)
1419 *r_firstblend = 1.0f;
1420 *r_lastblend = 1.0f;
1423 static void calc_bevfac_mapping(Curve *cu, BevList *bl, Nurb *nu,
1424 int *r_start, float *r_firstblend, int *r_steps, float *r_lastblend)
1426 float tmpf, total_length = 0.0f;
1429 if ((BKE_nurb_check_valid_u(nu) == false) ||
1430 /* not essential, but skips unnecessary calculation */
1431 (min_ff(cu->bevfac1, cu->bevfac2) == 0.0f &&
1432 max_ff(cu->bevfac1, cu->bevfac2) == 1.0f))
1434 calc_bevfac_mapping_default(bl, r_start, r_firstblend, r_steps, r_lastblend);
1438 if (ELEM(cu->bevfac1_mapping, CU_BEVFAC_MAP_SEGMENT, CU_BEVFAC_MAP_SPLINE) ||
1439 ELEM(cu->bevfac2_mapping, CU_BEVFAC_MAP_SEGMENT, CU_BEVFAC_MAP_SPLINE))
1441 for (i = 0; i < SEGMENTSU(nu); i++) {
1442 total_length += bl->seglen[i];
1446 switch (cu->bevfac1_mapping) {
1447 case CU_BEVFAC_MAP_RESOLU:
1449 const float start_fl = cu->bevfac1 * (bl->nr - 1);
1450 *r_start = (int)start_fl;
1451 *r_firstblend = 1.0f - (start_fl - (*r_start));
1454 case CU_BEVFAC_MAP_SEGMENT:
1456 calc_bevfac_segment_mapping(bl, cu->bevfac1, total_length, r_start, r_firstblend);
1457 *r_firstblend = 1.0f - *r_firstblend;
1460 case CU_BEVFAC_MAP_SPLINE:
1462 calc_bevfac_spline_mapping(bl, cu->bevfac1, total_length, r_start, r_firstblend);
1463 *r_firstblend = 1.0f - *r_firstblend;
1468 switch (cu->bevfac2_mapping) {
1469 case CU_BEVFAC_MAP_RESOLU:
1471 const float end_fl = cu->bevfac2 * (bl->nr - 1);
1474 *r_steps = 2 + end - *r_start;
1475 *r_lastblend = end_fl - end;
1478 case CU_BEVFAC_MAP_SEGMENT:
1480 calc_bevfac_segment_mapping(bl, cu->bevfac2, total_length, &end, r_lastblend);
1481 *r_steps = end - *r_start + 2;
1484 case CU_BEVFAC_MAP_SPLINE:
1486 calc_bevfac_spline_mapping(bl, cu->bevfac2, total_length, &end, r_lastblend);
1487 *r_steps = end - *r_start + 2;
1492 if (end < *r_start || (end == *r_start && *r_lastblend < 1.0f - *r_firstblend )) {
1493 SWAP(int, *r_start, end);
1494 tmpf = *r_lastblend;
1495 *r_lastblend = 1.0f - *r_firstblend;
1496 *r_firstblend = 1.0f - tmpf;
1497 *r_steps = end - *r_start + 2;
1500 if (*r_start + *r_steps > bl->nr) {
1501 *r_steps = bl->nr - *r_start;
1502 *r_lastblend = 1.0f;
1506 static void do_makeDispListCurveTypes(Scene *scene, Object *ob, ListBase *dispbase,
1507 DerivedMesh **r_dm_final,
1508 const bool for_render, const bool for_orco, const bool use_render_resolution)
1510 Curve *cu = ob->data;
1512 /* we do allow duplis... this is only displist on curve level */
1513 if (!ELEM(ob->type, OB_SURF, OB_CURVE, OB_FONT)) return;
1515 if (ob->type == OB_SURF) {
1516 BKE_displist_make_surf(scene, ob, dispbase, r_dm_final, for_render, for_orco, use_render_resolution);
1518 else if (ELEM(ob->type, OB_CURVE, OB_FONT)) {
1520 ListBase nubase = {NULL, NULL};
1522 BKE_curve_bevelList_free(&ob->curve_cache->bev);
1524 /* We only re-evlauate path if evaluation is not happening for orco.
1525 * If the calculation happens for orco, we should never free data which
1526 * was needed before and only not needed for orco calculation.
1529 if (ob->curve_cache->path) free_path(ob->curve_cache->path);
1530 ob->curve_cache->path = NULL;
1533 if (ob->type == OB_FONT) {
1534 BKE_vfont_to_curve_nubase(G.main, ob, FO_EDIT, &nubase);
1537 BKE_nurbList_duplicate(&nubase, BKE_curve_nurbs_get(cu));
1541 curve_calc_modifiers_pre(scene, ob, &nubase, for_render, use_render_resolution);
1543 BKE_curve_bevelList_make(ob, &nubase, for_render != false);
1545 /* If curve has no bevel will return nothing */
1546 BKE_curve_bevel_make(scene, ob, &dlbev, for_render, use_render_resolution);
1548 /* no bevel or extrude, and no width correction? */
1549 if (!dlbev.first && cu->width == 1.0f) {
1550 curve_to_displist(cu, &nubase, dispbase, for_render, use_render_resolution);
1553 float widfac = cu->width - 1.0f;
1554 BevList *bl = ob->curve_cache->bev.first;
1555 Nurb *nu = nubase.first;
1557 for (; bl && nu; bl = bl->next, nu = nu->next) {
1562 if (bl->nr) { /* blank bevel lists can happen */
1564 /* exception handling; curve without bevel or extrude, with width correction */
1565 if (BLI_listbase_is_empty(&dlbev)) {
1567 dl = MEM_callocN(sizeof(DispList), "makeDispListbev");
1568 dl->verts = MEM_mallocN(sizeof(float[3]) * bl->nr, "dlverts");
1569 BLI_addtail(dispbase, dl);
1571 if (bl->poly != -1) dl->type = DL_POLY;
1572 else dl->type = DL_SEGM;
1574 if (dl->type == DL_SEGM) dl->flag = (DL_FRONT_CURVE | DL_BACK_CURVE);
1578 dl->col = nu->mat_nr;
1579 dl->charidx = nu->charidx;
1581 /* dl->rt will be used as flag for render face and */
1582 /* CU_2D conflicts with R_NOPUNOFLIP */
1583 dl->rt = nu->flag & ~CU_2D;
1586 bevp = bl->bevpoints;
1589 data[0] = bevp->vec[0] + widfac * bevp->sina;
1590 data[1] = bevp->vec[1] + widfac * bevp->cosa;
1591 data[2] = bevp->vec[2];
1598 ListBase bottom_capbase = {NULL, NULL};
1599 ListBase top_capbase = {NULL, NULL};
1600 float bottom_no[3] = {0.0f};
1601 float top_no[3] = {0.0f};
1602 float firstblend = 0.0f, lastblend = 0.0f;
1603 int i, start, steps;
1605 if (nu->flagu & CU_NURB_CYCLIC) {
1606 calc_bevfac_mapping_default(bl,
1607 &start, &firstblend, &steps, &lastblend);
1610 if (fabsf(cu->bevfac2 - cu->bevfac1) < FLT_EPSILON) {
1614 calc_bevfac_mapping(cu, bl, nu, &start, &firstblend, &steps, &lastblend);
1617 for (dlb = dlbev.first; dlb; dlb = dlb->next) {
1618 BevPoint *bevp_first, *bevp_last;
1621 /* for each part of the bevel use a separate displblock */
1622 dl = MEM_callocN(sizeof(DispList), "makeDispListbev1");
1623 dl->verts = data = MEM_mallocN(sizeof(float[3]) * dlb->nr * steps, "dlverts");
1624 BLI_addtail(dispbase, dl);
1628 dl->flag = dlb->flag & (DL_FRONT_CURVE | DL_BACK_CURVE);
1629 if (dlb->type == DL_POLY) {
1630 dl->flag |= DL_CYCL_U;
1632 if ((bl->poly >= 0) && (steps != 2)) {
1633 dl->flag |= DL_CYCL_V;
1638 dl->col = nu->mat_nr;
1639 dl->charidx = nu->charidx;
1641 /* dl->rt will be used as flag for render face and */
1642 /* CU_2D conflicts with R_NOPUNOFLIP */
1643 dl->rt = nu->flag & ~CU_2D;
1645 dl->bevelSplitFlag = MEM_callocN(sizeof(*dl->bevelSplitFlag) * ((steps + 0x1F) >> 5),
1648 /* for each point of poly make a bevel piece */
1649 bevp_first = bl->bevpoints;
1650 bevp_last = &bl->bevpoints[bl->nr - 1];
1651 bevp = &bl->bevpoints[start];
1652 for (i = start, a = 0; a < steps; i++, bevp++, a++) {
1654 float *cur_data = data;
1656 if (cu->taperobj == NULL) {
1660 float len, taper_fac;
1662 if (cu->flag & CU_MAP_TAPER) {
1663 len = (steps - 3) + firstblend + lastblend;
1667 else if (a == steps - 1)
1670 taper_fac = ((float) a - (1.0f - firstblend)) / len;
1674 taper_fac = (float) i / len;
1677 taper_fac += (1.0f - firstblend) / len;
1678 else if (a == steps - 1)
1679 taper_fac -= (1.0f - lastblend) / len;
1682 fac = displist_calc_taper(scene, cu->taperobj, taper_fac);
1685 if (bevp->split_tag) {
1686 dl->bevelSplitFlag[a >> 5] |= 1 << (a & 0x1F);
1689 /* rotate bevel piece and write in data */
1690 if ((a == 0) && (bevp != bevp_last)) {
1691 rotateBevelPiece(cu, bevp, bevp + 1, dlb, 1.0f - firstblend, widfac, fac, &data);
1693 else if ((a == steps - 1) && (bevp != bevp_first) ) {
1694 rotateBevelPiece(cu, bevp, bevp - 1, dlb, 1.0f - lastblend, widfac, fac, &data);
1697 rotateBevelPiece(cu, bevp, NULL, dlb, 0.0f, widfac, fac, &data);
1700 if (cu->bevobj && (cu->flag & CU_FILL_CAPS) && !(nu->flagu & CU_NURB_CYCLIC)) {
1702 fillBevelCap(nu, dlb, cur_data - 3 * dlb->nr, &bottom_capbase);
1703 negate_v3_v3(bottom_no, bevp->dir);
1705 if (a == steps - 1) {
1706 fillBevelCap(nu, dlb, cur_data, &top_capbase);
1707 copy_v3_v3(top_no, bevp->dir);
1712 /* gl array drawing: using indices */
1713 displist_surf_indices(dl);
1716 if (bottom_capbase.first) {
1717 BKE_displist_fill(&bottom_capbase, dispbase, bottom_no, false);
1718 BKE_displist_fill(&top_capbase, dispbase, top_no, false);
1719 BKE_displist_free(&bottom_capbase);
1720 BKE_displist_free(&top_capbase);
1726 BKE_displist_free(&dlbev);
1729 if (!(cu->flag & CU_DEFORM_FILL)) {
1730 curve_to_filledpoly(cu, &nubase, dispbase);
1734 if ((cu->flag & CU_PATH) ||
1735 DAG_get_eval_flags_for_object(scene, ob) & DAG_EVAL_NEED_CURVE_PATH)
1737 calc_curvepath(ob, &nubase);
1742 BKE_nurbList_duplicate(&ob->curve_cache->deformed_nurbs, &nubase);
1743 curve_calc_modifiers_post(scene, ob, &nubase, dispbase, r_dm_final, for_render, use_render_resolution);
1746 if (cu->flag & CU_DEFORM_FILL && !ob->derivedFinal) {
1747 curve_to_filledpoly(cu, &nubase, dispbase);
1750 BKE_nurbList_free(&nubase);
1754 void BKE_displist_make_curveTypes(Scene *scene, Object *ob, const bool for_orco)
1758 /* The same check for duplis as in do_makeDispListCurveTypes.
1759 * Happens when curve used for constraint/bevel was converted to mesh.
1760 * check there is still needed for render displist and orco displists. */
1761 if (!ELEM(ob->type, OB_SURF, OB_CURVE, OB_FONT))
1764 BKE_object_free_derived_caches(ob);
1766 if (!ob->curve_cache) {
1767 ob->curve_cache = MEM_callocN(sizeof(CurveCache), "CurveCache for curve types");
1770 dispbase = &(ob->curve_cache->disp);
1772 do_makeDispListCurveTypes(scene, ob, dispbase, &ob->derivedFinal, 0, for_orco, 0);
1774 boundbox_displist_object(ob);
1777 void BKE_displist_make_curveTypes_forRender(Scene *scene, Object *ob, ListBase *dispbase,
1778 DerivedMesh **r_dm_final, const bool for_orco,
1779 const bool use_render_resolution)
1781 if (ob->curve_cache == NULL) {
1782 ob->curve_cache = MEM_callocN(sizeof(CurveCache), "CurveCache for Curve");
1785 do_makeDispListCurveTypes(scene, ob, dispbase, r_dm_final, true, for_orco, use_render_resolution);
1788 void BKE_displist_make_curveTypes_forOrco(struct Scene *scene, struct Object *ob, struct ListBase *dispbase)
1790 if (ob->curve_cache == NULL) {
1791 ob->curve_cache = MEM_callocN(sizeof(CurveCache), "CurveCache for Curve");
1794 do_makeDispListCurveTypes(scene, ob, dispbase, NULL, 1, 1, 1);
1797 /* add Orco layer to the displist object which has got derived mesh and return orco */
1798 float *BKE_displist_make_orco(Scene *scene, Object *ob, DerivedMesh *dm_final,
1799 const bool for_render,
1800 const bool use_render_resolution)
1804 if (dm_final == NULL)
1805 dm_final = ob->derivedFinal;
1807 if (!dm_final->getVertDataArray(dm_final, CD_ORCO)) {
1808 curve_calc_orcodm(scene, ob, dm_final, for_render, use_render_resolution);
1811 orco = dm_final->getVertDataArray(dm_final, CD_ORCO);
1814 orco = MEM_dupallocN(orco);
1820 void BKE_displist_minmax(ListBase *dispbase, float min[3], float max[3])
1827 for (dl = dispbase->first; dl; dl = dl->next) {
1828 tot = (dl->type == DL_INDEX3) ? dl->nr : dl->nr * dl->parts;
1830 for (a = 0; a < tot; a++, vert += 3) {
1831 minmax_v3v3_v3(min, max, vert);
1837 /* there's no geometry in displist, use zero-sized boundbox */
1843 /* this is confusing, there's also min_max_object, appplying the obmat... */
1844 static void boundbox_displist_object(Object *ob)
1846 if (ELEM(ob->type, OB_CURVE, OB_SURF, OB_FONT)) {
1847 /* Curve's BB is already calculated as a part of modifier stack,
1848 * here we only calculate object BB based on final display list.
1851 /* object's BB is calculated from final displist */
1853 ob->bb = MEM_callocN(sizeof(BoundBox), "boundbox");
1855 if (ob->derivedFinal) {
1856 DM_set_object_boundbox(ob, ob->derivedFinal);
1859 float min[3], max[3];
1861 INIT_MINMAX(min, max);
1862 BKE_displist_minmax(&ob->curve_cache->disp, min, max);
1863 BKE_boundbox_init_from_minmax(ob->bb, min, max);