Reimplement dummy-edge logic

This will also work for more complex structures such as
testscube2_mitSpitze_offen.obj.
This commit is contained in:
hasufell 2014-05-16 19:04:38 +02:00
parent 29f06e91f4
commit f62e280cea
No known key found for this signature in database
GPG Key ID: 220CD1C5BDEED020
2 changed files with 59 additions and 24 deletions

View File

@ -155,9 +155,11 @@ bool vec_normal(HE_vert const * const vert, vector *vec)
for (uint32_t i = 0; i < ec; i++) { for (uint32_t i = 0; i < ec; i++) {
vector new_vec; vector new_vec;
if (edge_array[i]->face) {
FACE_NORMAL(edge_array[i], &new_vec); FACE_NORMAL(edge_array[i], &new_vec);
ADD_VECTORS(vec, &new_vec, vec); ADD_VECTORS(vec, &new_vec, vec);
} }
}
/* normalize the result */ /* normalize the result */
NORMALIZE_VECTOR(vec, vec); NORMALIZE_VECTOR(vec, vec);
@ -272,7 +274,8 @@ HE_obj *parse_obj(char const * const obj_string)
{ {
uint32_t vc = 0, /* vertices count */ uint32_t vc = 0, /* vertices count */
fc = 0, /* face count */ fc = 0, /* face count */
ec = 0; /* edge count */ ec = 0, /* edge count */
dec = 0; /* dummy edge count */
char *string, char *string,
*str_ptr_space = NULL, /* for strtok */ *str_ptr_space = NULL, /* for strtok */
*str_ptr_newline = NULL, /* for strtok */ *str_ptr_newline = NULL, /* for strtok */
@ -324,6 +327,7 @@ HE_obj *parse_obj(char const * const obj_string)
vertices[vc].edge = NULL; vertices[vc].edge = NULL;
vertices[vc].edge_array = NULL; vertices[vc].edge_array = NULL;
vertices[vc].eac = 0; vertices[vc].eac = 0;
vertices[vc].dc = 0;
/* allocate color struct and set preliminary colors */ /* allocate color struct and set preliminary colors */
vertices[vc].col = malloc(sizeof(color)); vertices[vc].col = malloc(sizeof(color));
@ -379,6 +383,7 @@ HE_obj *parse_obj(char const * const obj_string)
edges[ec].face = &(faces[i]); edges[ec].face = &(faces[i]);
edges[ec].pair = NULL; /* preliminary */ edges[ec].pair = NULL; /* preliminary */
vertices[fv_arr_id].edge = &(edges[ec]); /* last one wins */ vertices[fv_arr_id].edge = &(edges[ec]); /* last one wins */
vertices[fv_arr_id].dummys = NULL; /* preliminary */
/* Skip j == 0 here, so we don't underrun the arrays, /* Skip j == 0 here, so we don't underrun the arrays,
* since we always look one edge back. The first edge * since we always look one edge back. The first edge
@ -430,34 +435,48 @@ HE_obj *parse_obj(char const * const obj_string)
edges[i].pair = edges[i].vert->edge_array[j]; edges[i].pair = edges[i].vert->edge_array[j];
edges[i].vert->edge_array[j] = NULL; edges[i].vert->edge_array[j] = NULL;
/* this is a trick to make sure the
* edge member of HE_vert is never
* a border-edge (unless there are only
* border edges), otherwise
* get_all_emanating_edges() would break
* for vertices that are at the edge
* of an open object */
edges[i].vert->edge = &(edges[i]);
pair_found = true; pair_found = true;
break; break;
} }
} }
if (!pair_found) { /* we have a border edge */ /* create dummy pair edge if we have a border edge */
/* add dummy edge, so get_all_emanating_edges() if (!pair_found) {
* does not break */ uint32_t *vert_dc = &(edges[i].next->vert->dc);
edges[ec + i].face = NULL;
edges[ec + i].next = NULL; REALLOC(edges[i].next->vert->dummys,
edges[ec + i].pair = &(edges[i]); sizeof(HE_edge*) * (*vert_dc + 1));
edges[ec + i].vert = edges[i].next->vert;
edges[i].pair = &(edges[ec + i]); /* NULL-face indicates border-edge */
edges[ec + dec].face = NULL;
/* we don't know this one yet */
edges[ec + dec].next = NULL;
/* set both pairs */
edges[ec + dec].pair = &(edges[i]);
edges[i].pair = &(edges[ec + dec]);
/* set vertex */
edges[ec + dec].vert = edges[i].next->vert;
/* add the dummy edge to the dummys array of the vertex */
edges[ec + dec].vert->dummys[*vert_dc] = &(edges[ec + dec]);
(*vert_dc)++;
dec++;
}
}
/* now we have to connect the dummy edges together */
for (uint32_t i = 0; i < dec; i++) { /* for all dummy edges */
/* vertex the dummy edge points to */
HE_vert *vert = edges[ec + i].pair->vert;
/* iterate over the dummy array */
for (uint32_t j = 0; j < vert->dc; j++) {
if (vert == vert->dummys[j]->vert)
edges[ec + i].next = vert->dummys[j];
j++;
} }
} }
/* don't need the edge array anymore */
for (uint32_t i = 0; i < vc; i++)
free(vertices[i].edge_array);
obj = (HE_obj*) malloc(sizeof(HE_obj)); obj = (HE_obj*) malloc(sizeof(HE_obj));
CHECK_PTR_VAL(obj); CHECK_PTR_VAL(obj);
@ -469,10 +488,14 @@ HE_obj *parse_obj(char const * const obj_string)
obj->vc = vc; obj->vc = vc;
obj->fc = fc; obj->fc = fc;
free(string);
for (uint32_t i = 0; i < fc; i++) for (uint32_t i = 0; i < fc; i++)
free(face_v[i]); free(face_v[i]);
free(face_v); free(face_v);
for (uint32_t i = 0; i < vc; i++) {
free(vertices[i].dummys);
free(vertices[i].edge_array);
}
free(string);
return obj; return obj;
} }

View File

@ -149,11 +149,23 @@ struct HE_vert {
*/ */
HE_edge **edge_array; HE_edge **edge_array;
/**
* Similar as the edge_array acceleration structure,
* except that it is used for connecting the
* dummy edges.
*/
HE_edge **dummys;
/** /**
* Element count of the edge_array. * Element count of the edge_array.
*/ */
uint32_t eac; uint32_t eac;
/**
* Element count of dummys.
*/
uint32_t dc;
/** /**
* Color of the vertex. * Color of the vertex.
*/ */