forked from vectorgraphics/asymptote
-
Notifications
You must be signed in to change notification settings - Fork 0
/
drawpath3.h
162 lines (125 loc) · 3.75 KB
/
drawpath3.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
/*****
* drawpath3.h
*
* Stores a path3 that has been added to a picture.
*****/
#ifndef DRAWPATH3_H
#define DRAWPATH3_H
#include "drawelement.h"
#include "path3.h"
namespace camp {
class drawPath3 : public drawElement {
protected:
const path3 g;
triple center;
bool straight;
prc::RGBAColour color;
bool invisible;
Interaction interaction;
triple Min,Max;
public:
drawPath3(path3 g, triple center, const pen& p, Interaction interaction) :
g(g), center(center), straight(g.piecewisestraight()), color(rgba(p)),
invisible(p.invisible()), interaction(interaction),
Min(g.min()), Max(g.max()) {}
drawPath3(const double* t, const drawPath3 *s) :
g(camp::transformed(t,s->g)), straight(s->straight),
color(s->color), invisible(s->invisible), interaction(s->interaction),
Min(g.min()), Max(g.max()) {
center=t*s->center;
}
virtual ~drawPath3() {}
bool is3D() {return true;}
void bounds(const double* t, bbox3& B) {
if(t != NULL) {
const path3 tg(camp::transformed(t,g));
B.add(tg.min());
B.add(tg.max());
} else {
B.add(Min);
B.add(Max);
}
}
void ratio(const double* t, pair &b, double (*m)(double, double), double,
bool &first) {
pair z;
if(t != NULL) {
const path3 tg(camp::transformed(t,g));
z=tg.ratio(m);
} else z=g.ratio(m);
if(first) {
b=z;
first=false;
} else b=pair(m(b.getx(),z.getx()),m(b.gety(),z.gety()));
}
bool write(prcfile *out, unsigned int *, double, groupsmap&);
void render(GLUnurbs*, double, const triple&, const triple&, double,
bool lighton, bool transparent);
drawElement *transformed(const double* t);
};
class drawNurbsPath3 : public drawElement {
protected:
size_t degree;
size_t n;
triple *controls;
double *weights;
double *knots;
prc::RGBAColour color;
bool invisible;
triple Min,Max;
#ifdef HAVE_GL
GLfloat *Controls;
GLfloat *Knots;
#endif
public:
drawNurbsPath3(const vm::array& g, const vm::array* knot,
const vm::array* weight, const pen& p) :
color(rgba(p)), invisible(p.invisible()) {
size_t weightsize=checkArray(weight);
string wrongsize="Inconsistent NURBS data";
n=checkArray(&g);
if(n == 0 || (weightsize != 0 && weightsize != n))
reportError(wrongsize);
controls=new(UseGC) triple[n];
size_t k=0;
for(size_t i=0; i < n; ++i)
controls[k++]=vm::read<triple>(g,i);
if(weightsize > 0) {
size_t k=0;
weights=new(UseGC) double[n];
for(size_t i=0; i < n; ++i)
weights[k++]=vm::read<double>(weight,i);
} else weights=NULL;
size_t nknots=checkArray(knot);
if(nknots <= n+1 || nknots > 2*n)
reportError(wrongsize);
degree=nknots-n-1;
run::copyArrayC(knots,knot,0,NoGC);
#ifdef HAVE_GL
Controls=NULL;
#endif
}
drawNurbsPath3(const double* t, const drawNurbsPath3 *s) :
degree(s->degree), n(s->n), weights(s->weights), knots(s->knots),
color(s->color), invisible(s->invisible) {
controls=new(UseGC) triple[n];
for(unsigned int i=0; i < n; ++i)
controls[i]=t*s->controls[i];
#ifdef HAVE_GL
Controls=NULL;
#endif
}
bool is3D() {return true;}
void bounds(const double* t, bbox3& b);
virtual ~drawNurbsPath3() {}
bool write(prcfile *out, unsigned int *, double, groupsmap&);
void displacement();
void ratio(const double* t, pair &b, double (*m)(double, double), double fuzz,
bool &first);
void render(GLUnurbs *nurb, double size2,
const triple& Min, const triple& Max,
double perspective, bool lighton, bool transparent);
drawElement *transformed(const double* t);
};
}
#endif