2011-07-29 14:32:40 -03:00
|
|
|
/*
|
|
|
|
* extrude.c - Perform the extrusion
|
|
|
|
*
|
|
|
|
* Written 2011 by Werner Almesberger
|
|
|
|
* Copyright 2011 by Werner Almesberger
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Known bugs:
|
|
|
|
*
|
|
|
|
* - negative x coordinates in shape produce overlapping faces
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <math.h>
|
2011-07-31 19:17:05 -03:00
|
|
|
#include <assert.h>
|
2011-07-29 14:32:40 -03:00
|
|
|
|
2011-07-31 19:17:05 -03:00
|
|
|
#include "ptrude.h"
|
2011-07-29 14:32:40 -03:00
|
|
|
#include "path.h"
|
|
|
|
#include "extrude.h"
|
|
|
|
|
|
|
|
|
2011-07-31 09:57:51 -03:00
|
|
|
static void cvt_3d(struct point *p, const struct vertex *v, double z)
|
2011-07-29 14:32:40 -03:00
|
|
|
{
|
2011-07-31 09:57:51 -03:00
|
|
|
p->x = v->x;
|
|
|
|
p->y = v->y;
|
|
|
|
p->z = z;
|
2011-07-29 14:32:40 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-07-31 19:17:05 -03:00
|
|
|
static void size(const struct path *p)
|
|
|
|
{
|
|
|
|
const struct vertex *v, *next;
|
|
|
|
double s = 0, sl = 0;
|
|
|
|
int n = 0;
|
|
|
|
|
|
|
|
if (!debug)
|
|
|
|
return;
|
|
|
|
for (v = p->vertices; v; v = next) {
|
|
|
|
next = v->next;
|
|
|
|
if (next)
|
|
|
|
s += hypot(v->x-next->x, v->y-next->y);
|
|
|
|
sl += v->len;
|
|
|
|
n++;
|
|
|
|
}
|
|
|
|
fprintf(stderr, "%d virt %g real %g\n", n, sl, s);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-07-31 09:57:51 -03:00
|
|
|
static void mesh(const struct path *pa, double za,
|
|
|
|
const struct path *pb, double zb,
|
2011-07-29 14:32:40 -03:00
|
|
|
void (*face)(void *data, struct point a, struct point b, struct point c),
|
|
|
|
void *data)
|
|
|
|
{
|
2011-07-31 09:57:51 -03:00
|
|
|
const struct vertex *a = pa->vertices;
|
|
|
|
const struct vertex *b = pb->vertices;
|
|
|
|
double sa = 0, sb = 0; /* total distance traveled on path */
|
|
|
|
double da, db;
|
|
|
|
struct point a3, b3, n;
|
|
|
|
|
|
|
|
cvt_3d(&a3, a, za);
|
|
|
|
cvt_3d(&b3, b, zb);
|
2011-07-31 19:17:05 -03:00
|
|
|
|
2011-07-31 09:57:51 -03:00
|
|
|
while (a->next || b->next) {
|
|
|
|
da = fabs(sa+a->len-sb);
|
|
|
|
db = fabs(sb+b->len-sa);
|
|
|
|
if ((a->next && da < db) || !b->next) {
|
|
|
|
cvt_3d(&n, a->next, za);
|
|
|
|
face(data, a3, n, b3);
|
|
|
|
a3 = n;
|
|
|
|
sa += a->len;
|
|
|
|
a = a->next;
|
|
|
|
} else {
|
|
|
|
cvt_3d(&n, b->next, zb);
|
|
|
|
face(data, b3, a3, n);
|
|
|
|
b3 = n;
|
|
|
|
sb += b->len;
|
|
|
|
b = b->next;
|
|
|
|
}
|
2011-07-29 14:32:40 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void extrude(const struct path *path, const struct path *shape,
|
2011-07-31 09:57:51 -03:00
|
|
|
double r, double d,
|
2011-07-29 14:32:40 -03:00
|
|
|
void (*face)(void *data, struct point a, struct point b, struct point c),
|
|
|
|
void *data)
|
|
|
|
{
|
2011-07-31 09:57:51 -03:00
|
|
|
const struct vertex *v, *next; /* current and next vertex */
|
|
|
|
struct path *prev, *curr; /* previous and current path */
|
|
|
|
struct path *tmp;
|
2011-07-29 14:32:40 -03:00
|
|
|
|
2011-07-31 09:57:51 -03:00
|
|
|
if (!path->vertices || !path->vertices->next)
|
|
|
|
return;
|
2011-07-29 14:32:40 -03:00
|
|
|
|
2011-07-31 09:57:51 -03:00
|
|
|
v = shape->vertices;
|
|
|
|
if (!v || !v->next)
|
|
|
|
return;
|
2011-07-29 14:32:40 -03:00
|
|
|
|
2011-07-31 19:17:05 -03:00
|
|
|
tmp = stretch_path(path, v->x, r);
|
|
|
|
size(tmp);
|
2011-07-31 09:57:51 -03:00
|
|
|
prev = round_path(tmp, r, d);
|
|
|
|
free_path(tmp);
|
2011-07-31 19:17:05 -03:00
|
|
|
size(prev);
|
2011-07-29 14:32:40 -03:00
|
|
|
|
2011-07-31 09:57:51 -03:00
|
|
|
while (v->next) {
|
|
|
|
next = v->next;
|
2011-07-31 19:17:05 -03:00
|
|
|
tmp = stretch_path(path, next->x, r);
|
|
|
|
size(tmp);
|
2011-07-31 09:57:51 -03:00
|
|
|
curr = round_path(tmp, r, d);
|
2011-07-31 19:17:05 -03:00
|
|
|
size(curr);
|
2011-07-31 09:57:51 -03:00
|
|
|
free_path(tmp);
|
2011-07-29 14:32:40 -03:00
|
|
|
|
2011-07-31 09:57:51 -03:00
|
|
|
mesh(prev, v->y, curr, next->y, face, data);
|
2011-07-29 14:32:40 -03:00
|
|
|
|
2011-07-31 09:57:51 -03:00
|
|
|
free_path(prev);
|
|
|
|
prev = curr;
|
|
|
|
v = next;
|
2011-07-29 14:32:40 -03:00
|
|
|
}
|
|
|
|
|
2011-07-31 09:57:51 -03:00
|
|
|
free_path(prev);
|
2011-07-29 14:32:40 -03:00
|
|
|
}
|