JunkBox_Lib++ (for Windows) 1.10.1
Loading...
Searching...
No Matches
AffineTrans.h
Go to the documentation of this file.
1#ifndef __JBXL_CPP_AFFINETRANS_H_
2#define __JBXL_CPP_AFFINETRANS_H_
3
10#include "Rotation.h"
11
12
13//
14namespace jbxl {
15
16
18// Affine Transfer
19//
20
27template <typename T=double> class DllExport AffineTrans
28{
29private:
30 Matrix<T> matrix; // Matrix(2, 4, 4); 2次元 4x4行列
31
35
36private:
39
40public:
41 AffineTrans(void) { init();}
42 virtual ~AffineTrans(void) {}
43
44 void init(void) { initComponents(); matrix = Matrix<T>(2, 4, 4); _changed_matrix = false; computeMatrix();}
45 void setup(void){ init();}
46 void initComponents(void) { initScale(); initRotation(); initShift();}
47
48 void set(Vector<T> s, Quaternion<T> q, Vector<T> t) { scale = s; shift = t, rotate = q; computeMatrix();}
49 void free(void) { initComponents(); matrix.free(); _changed_matrix = false; computeMatrix();}
50 void clear(void){ initComponents(); matrix.clear(); _changed_matrix = false; computeMatrix();}
53
54 void clean_matrix(void) { _changed_matrix = false;}
55 void clean_components(void) { _changed_components = false;}
56 void changed_matrix(void) { _changed_matrix = true;}
57 void changed_components(void) { _changed_components = true;}
58 bool is_changed_matrix(void) { return _changed_matrix;} // true なら computeComponents() を実行する(推奨)
59 bool is_changed_components(void) { return _changed_components;} // true なら computeMatrix() を実行する(推奨)
60 bool need_compute_matrix(void) { return ( _changed_components && !_changed_matrix);}
61 bool need_compute_components(void){ return (!_changed_components && _changed_matrix);}
62
63 void computeMatrix(bool with_scale = true);
65
66 void initShift(void) { shift.init(); _changed_components = true;}
67 void initScale(void) { scale.set((T)1.0, (T)1.0, (T)1.0); _changed_components = true;}
68 void initRotation(void) { rotate.init(); _changed_components = true;}
69
70 void setShift(T x, T y, T z) { shift.set(x, y, z); _changed_components = true;}
71 void setScale(T x, T y, T z) { scale.set(x, y, z); _changed_components = true;}
72 void setRotation(T s, T x, T y, T z) { rotate.set(s, x, y, z); _changed_components = true;}
73
74 void setShift(Vector<T> v) { shift = v; _changed_components = true;}
75 void setScale(Vector<T> v) { scale = v; _changed_components = true;}
76 void setRotation(Quaternion<T> q) { rotate = q; _changed_components = true;}
77
78 void addShift(T x, T y, T z) { shift.x += x; shift.y += y; shift.z += z; _changed_components = true;}
79 void addScale(T x, T y, T z) { scale.x *= x; scale.y *= y; scale.z *= z; _changed_components = true;}
80 void addRotation(T s, T x, T y, T z) { Quaternion<T> q(s, x, y, z); rotate = q*rotate; _changed_components = true;}
81
82 void addShift(Vector<T> v) { shift = shift + v; _changed_components = true;}
83 void addScale(Vector<T> v) { scale = shift + v; _changed_components = true;}
84 void addRotation(Quaternion<T> q) { rotate = q*rotate; _changed_components = true;}
85
86 void setMatrix(int i, int j, T v) { matrix.element(i, j) = v; _changed_matrix = true;}
87 T getMatrix(int i, int j) { return matrix.element(i, j);}
88 void element(int i, int j, T v) { matrix.element(i, j) = v; _changed_matrix = true;}
89 T element(int i, int j) { return matrix.element(i, j);}
90
91 T getShiftX(void) { return shift.x;}
92 T getShiftY(void) { return shift.y;}
93 T getShiftZ(void) { return shift.z;}
94 T getScaleX(void) { return scale.x;}
95 T getScaleY(void) { return scale.y;}
96 T getScaleZ(void) { return scale.z;}
97 T getRotationS(void){ return rotate.s;}
98 T getRotationX(void){ return rotate.x;}
99 T getRotationY(void){ return rotate.y;}
100 T getRotationZ(void){ return rotate.z;}
101
102 Vector<T> getScale (void) { return scale;}
103 Quaternion<T> getRotation(void) { return rotate;}
104 Vector<T> getShift (void) { return shift;}
105 Matrix<T> getMatrix(void) { return matrix;}
108
109 // operator * は コンポーネントが計算されていることが条件.
110 // 下記関数は コンポ―テントが計算されていなくても良い.
111 void affineMatrixAfter (AffineTrans<T> a); // *this = (*this) * a を Matrix のままで計算する.
112 void affineMatrixBefore(AffineTrans<T> a); // *this = a * (*this) を Matrix のままで計算する.
113
114 bool isSetComponents(void) { if(isSetShift() || isSetScale() || isSetRotation()) return true; else return false;}
115 bool isSetShift(void){ return (shift !=Vector<T>());}
116 bool isSetScale(void){ return (scale !=Vector<T>((T)1.0, (T)1.0, (T)1.0));}
117 bool isSetRotation(void) { return (rotate!=Quaternion<T>());}
118 bool isNormal(void){ if(scale.x>=JBXL_EPS && scale.y>=JBXL_EPS && scale.z>=JBXL_EPS) return true; else return false;}
119
121
122 Vector<T> execTrans(Vector<T> v) { return execShift(execRotation(execScale(v)));}
123 Vector<T> execInvTrans(Vector<T> v) { return execInvScale(execInvRotation(execInvShift(v)));}
124
125 Vector<T> execRotationScale(Vector<T> v) { return execRotation(execScale(v));}
126 Vector<T> execInvRotationScale(Vector<T> v) { return execInvScale(execInvRotation(v));}
127 //
128 Vector<T> execShift(Vector<T> v) { return Vector<T>(v.x+shift.x, v.y+shift.y, v.z+shift.z, (T)0.0, Min(v.c, shift.c));}
129 Vector<T> execInvShift(Vector<T> v) { return Vector<T>(v.x-shift.x, v.y-shift.y, v.z-shift.z, (T)0.0, Min(v.c, shift.c));}
130 Vector<T> execScale(Vector<T> v) { return Vector<T>(v.x*scale.x, v.y*scale.y, v.z*scale.z, (T)0.0, Min(v.c, scale.c));}
131 Vector<T> execInvScale(Vector<T> v) { return Vector<T>(v.x/scale.x, v.y/scale.y, v.z/scale.z, (T)0.0, Min(v.c, scale.c));}
134
135 // for debug
136 void printMatrix(void);
137 void printComponents(void);
138};
139
140
141template <typename T> inline void freeAffineTrans(AffineTrans<T>*& affine)
142{
143 if (affine!=NULL){
144 affine->free();
145 delete(affine);
146 affine = NULL;
147 }
148}
149
150
151template <typename T> inline AffineTrans<T>* newAffineTrans(AffineTrans<T> p)
152{
154 a->dup(p);
155 return a;
156}
157
158
163template <typename T> inline AffineTrans<T> operator * (AffineTrans<T> a, AffineTrans<T> b)
164{
165 a.computeMatrix();
166 b.computeMatrix();
167
168 AffineTrans<T> affine;
169 for (int i=1; i<=4; i++) {
170 for (int j=1; j<=4; j++) {
171 T sum = (T)0.0;
172 for (int k=1; k<=4; k++) {
173 sum += a.element(i, k) * b.element(k, j);
174 }
175 affine.element(i, j, sum);
176 }
177 }
178 affine.computeComponents();
179
180 return affine;
181}
182
183
184
186// Affine Transfer
187//
188
189// AffineTrans のコピーを作る.マトリックスデータのメモリ部は共有しない.
190template <typename T> void AffineTrans<T>::dup(AffineTrans<T> a)
191{
192 *this = a;
193 matrix.dup(a.matrix);
194
195 return;
196}
197
198
199// 自分自身の コピーを作って返す.
200template <typename T> AffineTrans<T> AffineTrans<T>::dup(void)
201{
202 AffineTrans<T> affine;
203 affine.dup(*this);
204
205 return affine;
206}
207
208
209// 3x3 の回転行列を得る
210// Matrix が計算されていること.
212{
213 Matrix<T> mt;
214
215 if (matrix.element(4, 4)==(T)1.0) {
216 mt.init(2, 3, 3);
217 for (int j=1; j<=3; j++) {
218 for (int i=1; i<=3; i++) {
219 mt.element(i, j) = matrix.element(i, j);
220 }
221 }
222 }
223 else {
224 mt = rotate.getRotMatrix();
225 }
226
227 return mt;
228}
229
230
231// 各コンポーテントの成分から,逆変換を定義する.
233{
234 AffineTrans<T> affine;
235 if (!isNormal()) return affine;
236 computeComponents();
237
238 Matrix<T> rsz(2, 3, 3);
239 rsz.element(1, 1) = (T)1.0/scale.x;
240 rsz.element(2, 2) = (T)1.0/scale.y;
241 rsz.element(3, 3) = (T)1.0/scale.z;
242
243 Matrix<T> rmt = rsz*(~rotate).getRotMatrix();
244
245 affine.matrix.element(4, 4) = (T)1.0;
246 for (int j=1; j<=3; j++) {
247 for (int i=1; i<=3; i++) {
248 affine.matrix.element(i, j) = rmt.element(i, j);
249 }
250 }
251 rsz.free();
252 rmt.free();
253
254 Matrix<T> rst(2, 4, 4);
255 rst.element(1, 1) = rst.element(2,2) = rst.element(3,3) = rst.element(4,4) = (T)1.0;
256 rst.element(1, 4) = -shift.x;
257 rst.element(2, 4) = -shift.y;
258 rst.element(3, 4) = -shift.z;
259
260 affine.matrix = affine.matrix*rst;
261 affine.computeComponents();
262
263 rst.free();
264
265 return affine;
266}
267
268
275{
276 AffineTrans<T> affine;
277 for (int i=1; i<=4; i++) {
278 for (int j=1; j<=4; j++) {
279 T sum = (T)0;
280 for (int k=1; k<=4; k++) {
281 sum += matrix.element(i, k) * a.element(k, j);
282 }
283 affine.element(i, j, sum);
284 }
285 }
286 for (int i=1; i<=4; i++) {
287 for (int j=1; j<=4; j++) {
288 matrix.element(i, j) = affine.element(i, j);
289 }
290 }
291 affine.free();
292 computeComponents();
293 return;
294}
295
296
303{
304 AffineTrans<T> affine;
305 for (int i=1; i<=4; i++) {
306 for (int j=1; j<=4; j++) {
307 T sum = (T)0;
308 for (int k=1; k<=4; k++) {
309 sum += a.element(i, k) * matrix.element(k, j);
310 }
311 affine.element(i, j, sum);
312 }
313 }
314 for (int i=1; i<=4; i++) {
315 for (int j=1; j<=4; j++) {
316 matrix.element(i, j) = affine.element(i, j);
317 }
318 }
319 affine.free();
320 computeComponents();
321 return;
322}
323
324
325// 各コンポーネントから,変換行列を計算する.
326template <typename T> void AffineTrans<T>::computeMatrix(bool with_scale)
327{
328 if (_changed_matrix) {
329 PRINT_MESG("AffineTrans<T>::computeMatrix: ERROR: may destroy Matrix!\n");
330 }
331
332 Matrix<T> sz(2, 3, 3);
333 if (with_scale) {
334 sz.element(1, 1) = scale.x;
335 sz.element(2, 2) = scale.y;
336 sz.element(3, 3) = scale.z;
337 }
338 else {
339 sz.element(1, 1) = (T)1.0;
340 sz.element(2, 2) = (T)1.0;
341 sz.element(3, 3) = (T)1.0;
342 }
343
344 matrix.clear();
345 Matrix<T> mt = rotate.getRotMatrix()*sz;
346 for (int j=1; j<=3; j++) {
347 for (int i=1; i<=3; i++) {
348 matrix.element(i, j) = mt.element(i, j);
349 }
350 }
351 sz.free();
352 mt.free();
353
354 matrix.element(1, 4) = shift.x;
355 matrix.element(2, 4) = shift.y;
356 matrix.element(3, 4) = shift.z;
357 matrix.element(4, 4) = (T)1.0;
358 matrix.element(4, 1) = (T)0.0;
359 matrix.element(4, 2) = (T)0.0;
360 matrix.element(4, 3) = (T)0.0;
361
362 _changed_matrix = false;
363 _changed_components = false;
364 return;
365}
366
367
368// 変換行列から,各コンポーネントを再計算する.
369template <typename T> void AffineTrans<T>::computeComponents(void)
370{
371 if (_changed_components) {
372 PRINT_MESG("AffineTrans<T>::computeComponents: ERROR: may destroy Components!\n");
373 }
374
375 T sx, sy, sz;
376 sx = sy = sz = (T)0.0;
377 for (int i=1; i<=3; i++) {
378 sx += matrix.element(i, 1)*matrix.element(i, 1);
379 sy += matrix.element(i, 2)*matrix.element(i, 2);
380 sz += matrix.element(i, 3)*matrix.element(i, 3);
381 }
382 sx = (T)sqrt(sx);
383 sy = (T)sqrt(sy);
384 sz = (T)sqrt(sz);
385 if (!isNormal()) return;
386 scale.set(sx, sy, sz);
387
388 //
389 Matrix<T> mt = getRotationMatrix();
390
391 for (int i=1; i<=3; i++) {
392 mt.element(i, 1) /= sx;
393 mt.element(i, 2) /= sy;
394 mt.element(i, 3) /= sz;
395 }
396 rotate = RotMatrix2Quaternion<T>(mt);
397 mt.free();
398 //
399 shift.set(matrix.element(1,4), matrix.element(2,4), matrix.element(3,4));
400
401 _changed_matrix = false;
402 _changed_components = false;
403 return;
404}
405
406
407// 変換行列を用いて,ベクトルを変換する.
409{
410 //if (matrix.element(4,4)!=(T)1.0) computeMatrix(true);
411 computeMatrix(true);
412
413 Matrix<T> mv(1, 4);
414 mv.mx[0] = v.x;
415 mv.mx[1] = v.y;
416 mv.mx[2] = v.z;
417 mv.mx[3] = (T)1.0;
418
419 Matrix<T> mt = matrix * mv;
420 Vector<T> vct;
421 vct.x = mt.mx[0];
422 vct.y = mt.mx[1];
423 vct.z = mt.mx[2];
424
425 mt.free();
426 mv.free();
427
428 return vct;
429}
430
431
432template <typename T> void AffineTrans<T>::printMatrix(void)
433{
434 for(int j=1; j<=4; j++) {
435 for(int i=1; i<=4; i++) {
436 T element = matrix.element(i, j);
437 print_message(" %g ", element);
438 }
439 print_message("\n");
440 }
441 return;
442}
443
444
445template <typename T> void AffineTrans<T>::printComponents(void)
446{
447 print_message("shift = (%g, %g, %g)\n", shift.x, shift.y, shift.z);
448 print_message("scale = (%g, %g, %g)\n", scale.x, scale.y, scale.z);
449 print_message("rot = (%g, %g, %g, %g)\n", rotate.s, rotate.x, rotate.y, rotate.z);
450}
451
452
453
454} // namespace
455
456
457#endif
458
回転・クォータニオン ライブラリ ヘッダ
void setShift(T x, T y, T z)
Definition AffineTrans.h:70
Matrix< T > matrix
Definition AffineTrans.h:30
void computeMatrix(bool with_scale=true)
void dup(AffineTrans a)
void set(Vector< T > s, Quaternion< T > q, Vector< T > t)
Definition AffineTrans.h:48
Vector< T > execInvShift(Vector< T > v)
bool isSetRotation(void)
AffineTrans< T > getInverseAffine(void)
void addRotation(Quaternion< T > q)
Definition AffineTrans.h:84
void init(void)
Definition AffineTrans.h:44
Quaternion< T > rotate
Definition AffineTrans.h:34
void initRotation(void)
Definition AffineTrans.h:68
void setShift(Vector< T > v)
Definition AffineTrans.h:74
Vector< T > execInvTrans(Vector< T > v)
Vector< T > getScale(void)
bool isSetComponents(void)
void affineMatrixAfter(AffineTrans< T > a)
Vector< T > execInvScale(Vector< T > v)
void setScale(T x, T y, T z)
Definition AffineTrans.h:71
Matrix< T > getMatrix(void)
bool isNormal(void)
Matrix< T > getRotationMatrix(void)
void printMatrix(void)
Vector< T > execRotationScale(Vector< T > v)
void printComponents(void)
void initScale(void)
Definition AffineTrans.h:67
Vector< T > scale
Definition AffineTrans.h:33
bool need_compute_components(void)
Definition AffineTrans.h:61
T element(int i, int j)
Definition AffineTrans.h:89
void initComponents(void)
Definition AffineTrans.h:46
void setRotation(Quaternion< T > q)
Definition AffineTrans.h:76
void changed_matrix(void)
Definition AffineTrans.h:56
Vector< T > execShift(Vector< T > v)
bool is_changed_components(void)
Definition AffineTrans.h:59
void setup(void)
Definition AffineTrans.h:45
Vector< T > execScale(Vector< T > v)
Quaternion< T > getRotation(void)
bool isSetScale(void)
void computeComponents(void)
Vector< T > execInvRotationScale(Vector< T > v)
void addRotation(T s, T x, T y, T z)
Definition AffineTrans.h:80
void setScale(Vector< T > v)
Definition AffineTrans.h:75
Vector< T > execTrans(Vector< T > v)
bool is_changed_matrix(void)
Definition AffineTrans.h:58
void addScale(T x, T y, T z)
Definition AffineTrans.h:79
void changed_components(void)
Definition AffineTrans.h:57
T getMatrix(int i, int j)
Definition AffineTrans.h:87
bool isSetShift(void)
void element(int i, int j, T v)
Definition AffineTrans.h:88
void setMatrix(int i, int j, T v)
Definition AffineTrans.h:86
Vector< T > shift
Definition AffineTrans.h:32
void initShift(void)
Definition AffineTrans.h:66
Vector< T > execInvRotation(Vector< T > v)
void affineMatrixBefore(AffineTrans< T > a)
void free(void)
Definition AffineTrans.h:49
void addShift(Vector< T > v)
Definition AffineTrans.h:82
void addScale(Vector< T > v)
Definition AffineTrans.h:83
void clean_matrix(void)
Definition AffineTrans.h:54
virtual ~AffineTrans(void)
Definition AffineTrans.h:42
Vector< T > execRotation(Vector< T > v)
void addShift(T x, T y, T z)
Definition AffineTrans.h:78
AffineTrans< T > dup(void)
void clean_components(void)
Definition AffineTrans.h:55
bool need_compute_matrix(void)
Definition AffineTrans.h:60
void clear(void)
Definition AffineTrans.h:50
Vector< T > execMatrixTrans(Vector< T > v)
Vector< T > getShift(void)
void setRotation(T s, T x, T y, T z)
Definition AffineTrans.h:72
void init()
Definition Matrix++.h:43
T & element(int i,...)
Definition Matrix++.h:204
void free()
free() は手動で呼び出す.
Definition Matrix++.h:52
T * mx
要素 mx[0] 〜 mx[r-1]
Definition Matrix++.h:35
T y
y 成分
Definition Rotation.h:48
void set(T S, T X, T Y, T Z, T N=(T) 0.0, T C=(T) 1.0)
Definition Rotation.h:273
void init(T C=(T) 1.0)
Definition Rotation.h:64
T x
x 成分
Definition Rotation.h:47
T z
z 成分
Definition Rotation.h:49
T s
cos(θ/2)
Definition Rotation.h:46
void init(double C=1.0)
Definition Vector.h:74
double c
信頼度
Definition Vector.h:63
void set(T X, T Y=0, T Z=0, double N=0.0, double C=1.0, int D=0)
Definition Vector.h:110
#define Min(x, y)
Definition common.h:250
#define JBXL_EPS
Definition common.h:295
#define DllExport
Definition common.h:105
Definition Brep.h:29
AffineTrans< T > operator*(AffineTrans< T > a, AffineTrans< T > b)
Vector< T > VectorRotation(Vector< T > v, Quaternion< T > q)
Definition Rotation.h:1187
AffineTrans< T > * newAffineTrans(AffineTrans< T > p)
Vector< T > VectorInvRotation(Vector< T > v, Quaternion< T > q)
Definition Rotation.h:1207
void freeAffineTrans(AffineTrans< T > *&affine)
void print_message(const char *fmt,...)
バッファリングなしのメッセージ出力(stderr)
Definition tools.cpp:4054
#define PRINT_MESG(...)
環境依存用の出力関数.MS Windows用は未実装
Definition tools.h:469