1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-04 18:06:22 +03:00
occt/src/gp/gp_Trsf.cxx
jgv bead40f28c 0023774: Incorrect conversion from gp_Trsf2d to gp_Trsf
Adding test command for this fix
Adding test case for this fix
2013-03-15 12:40:54 +04:00

824 lines
24 KiB
C++
Executable File

// Copyright (c) 1995-1999 Matra Datavision
// Copyright (c) 1999-2012 OPEN CASCADE SAS
//
// The content of this file is subject to the Open CASCADE Technology Public
// License Version 6.5 (the "License"). You may not use the content of this file
// except in compliance with the License. Please obtain a copy of the License
// at http://www.opencascade.org and read it completely before using this file.
//
// The Initial Developer of the Original Code is Open CASCADE S.A.S., having its
// main offices at: 1, place des Freres Montgolfier, 78280 Guyancourt, France.
//
// The Original Code and all software distributed under the License is
// distributed on an "AS IS" basis, without warranty of any kind, and the
// Initial Developer hereby disclaims all such warranties, including without
// limitation, any warranties of merchantability, fitness for a particular
// purpose or non-infringement. Please see the License for the specific terms
// and conditions governing the rights and limitations under the License.
// JCV 30/08/90 Modif passage version C++ 2.0 sur Sun
// JCV 1/10/90 Changement de nom du package vgeom -> gp
// JCV 4/10/90 codage sur la forme de la transformation shape,Scaling,negative
// JCV 10/12/90 Modif introduction des classes Mat et XYZ dans gp
#define No_Standard_OutOfRange
#include <gp_Trsf.ixx>
#include <gp.hxx>
#include <Standard_ConstructionError.hxx>
//=======================================================================
//function : gp_Trsf
//purpose : Constructor from 2d
//=======================================================================
gp_Trsf::gp_Trsf (const gp_Trsf2d& T) :
scale(T.ScaleFactor()),
shape(T.Form()),
loc(T.TranslationPart().X(),T.TranslationPart().Y(), 0.0)
{
const gp_Mat2d& M = T.HVectorialPart();
matrix(1,1) = M(1,1);
matrix(1,2) = M(1,2);
matrix(2,1) = M(2,1);
matrix(2,2) = M(2,2);
matrix(3,3) = 1.;
if (shape == gp_Ax1Mirror)
{
scale = 1;
matrix.Multiply(-1);
}
}
//=======================================================================
//function : SetMirror
//purpose :
//=======================================================================
void gp_Trsf::SetMirror (const gp_Ax1& A1)
{
shape = gp_Ax1Mirror;
scale = 1;
loc = A1.Location().XYZ();
matrix.SetDot(A1.Direction().XYZ());
matrix.Multiply(-2);
matrix.SetDiagonal (matrix.Value (1,1) + 1,
matrix.Value (2,2) + 1,
matrix.Value (3,3) + 1);
loc.Multiply (matrix);
loc.Add (A1.Location().XYZ());
matrix.Multiply(-1);
}
//=======================================================================
//function : SetMirror
//purpose :
//=======================================================================
void gp_Trsf::SetMirror (const gp_Ax2& A2)
{
shape = gp_Ax2Mirror;
scale = -1;
loc = A2.Location().XYZ();
matrix.SetDot(A2.Direction().XYZ());
matrix.Multiply(2);
matrix.SetDiagonal (matrix.Value (1,1) - 1,
matrix.Value (2,2) - 1,
matrix.Value (3,3) - 1);
loc.Multiply (matrix);
loc.Add (A2.Location().XYZ());
}
//=======================================================================
//function : SetRotation
//purpose :
//=======================================================================
void gp_Trsf::SetRotation (const gp_Ax1& A1,
const Standard_Real Ang)
{
shape = gp_Rotation;
scale = 1.;
loc = A1.Location().XYZ();
matrix.SetRotation (A1.Direction().XYZ(), Ang);
loc.Reverse ();
loc.Multiply (matrix);
loc.Add (A1.Location().XYZ());
}
//=======================================================================
//function : SetRotation
//purpose :
//=======================================================================
void gp_Trsf::SetRotation (const gp_Quaternion& R)
{
shape = gp_Rotation;
scale = 1.;
loc.SetCoord (0., 0., 0.);
matrix = R.GetMatrix();
}
//=======================================================================
//function : SetScale
//purpose :
//=======================================================================
void gp_Trsf::SetScale (const gp_Pnt& P, const Standard_Real S)
{
shape = gp_Scale;
scale = S;
loc = P.XYZ();
Standard_Real As = scale;
if (As < 0) As = - As;
Standard_ConstructionError_Raise_if
(As <= gp::Resolution(),"gp_Trsf::SetScaleFactor");
matrix.SetIdentity ();
loc.Multiply (1-S);
}
//=======================================================================
//function : SetTransformation
//purpose :
//=======================================================================
void gp_Trsf::SetTransformation (const gp_Ax3& FromA1,
const gp_Ax3& ToA2)
{
shape = gp_CompoundTrsf;
scale = 1.0;
// matrix from XOY ToA2 :
matrix.SetCol (1, ToA2.XDirection().XYZ());
matrix.SetCol (2, ToA2.YDirection().XYZ());
matrix.SetCol (3, ToA2.Direction().XYZ());
loc = ToA2.Location().XYZ();
matrix.Transpose();
loc.Multiply (matrix);
loc.Reverse ();
// matrix FromA1 to XOY :
const gp_XYZ& xDir = FromA1.XDirection().XYZ();
const gp_XYZ& yDir = FromA1.YDirection().XYZ();
const gp_XYZ& zDir = FromA1.Direction().XYZ();
gp_Mat MA1 (xDir, yDir, zDir);
gp_XYZ MA1loc = FromA1.Location().XYZ();
// matrix * MA1 => FromA1 ToA2 :
MA1loc.Multiply (matrix);
loc.Add (MA1loc);
matrix.Multiply (MA1);
}
void gp_Trsf::SetTransformation (const gp_Ax3& A3)
{
shape = gp_CompoundTrsf;
scale = 1.0;
loc = A3.Location().XYZ();
matrix.SetCols (A3.XDirection().XYZ(),
A3.YDirection().XYZ(),
A3. Direction().XYZ());
matrix.Transpose();
loc.Multiply (matrix);
loc.Reverse ();
}
//=======================================================================
//function : SetTransformation
//purpose :
//=======================================================================
void gp_Trsf::SetTransformation (const gp_Quaternion& R, const gp_Vec& T)
{
shape = gp_CompoundTrsf;
scale = 1.;
loc = T.XYZ();
matrix = R.GetMatrix();
}
//=======================================================================
//function : SetDisplacement
//purpose :
//=======================================================================
void gp_Trsf::SetDisplacement (const gp_Ax3& FromA1,
const gp_Ax3& ToA2)
{
shape = gp_CompoundTrsf;
scale = 1.0;
// matrix from ToA2 to XOY :
matrix.SetCol (1, ToA2.XDirection().XYZ());
matrix.SetCol (2, ToA2.YDirection().XYZ());
matrix.SetCol (3, ToA2.Direction().XYZ());
loc = ToA2.Location().XYZ();
// matrix XOY to FromA1 :
const gp_XYZ& xDir = FromA1.XDirection().XYZ();
const gp_XYZ& yDir = FromA1.YDirection().XYZ();
const gp_XYZ& zDir = FromA1.Direction().XYZ();
gp_Mat MA1 (xDir, yDir, zDir);
MA1.Transpose();
gp_XYZ MA1loc = FromA1.Location().XYZ();
MA1loc.Multiply (MA1);
MA1loc.Reverse();
// matrix * MA1
MA1loc.Multiply (matrix);
loc.Add (MA1loc);
matrix.Multiply (MA1);
}
//=======================================================================
//function : SetTranslationPart
//purpose :
//=======================================================================
void gp_Trsf::SetTranslationPart (const gp_Vec& V) {
loc = V.XYZ();
Standard_Real X = loc.X();
if (X < 0) X = - X;
Standard_Real Y = loc.Y();
if (Y < 0) Y = - Y;
Standard_Real Z = loc.Z();
if (Z < 0) Z = - Z;
Standard_Boolean locnull =
(X <= gp::Resolution() &&
Y <= gp::Resolution() &&
Z <= gp::Resolution());
switch (shape) {
case gp_Identity :
if (!locnull) shape = gp_Translation;
break;
case gp_Translation :
if (locnull) shape = gp_Identity;
break;
case gp_Rotation :
case gp_PntMirror :
case gp_Ax1Mirror :
case gp_Ax2Mirror :
case gp_Scale :
case gp_CompoundTrsf :
case gp_Other :
break;
}
}
//=======================================================================
//function : SetScaleFactor
//purpose :
//=======================================================================
void gp_Trsf::SetScaleFactor (const Standard_Real S)
{
Standard_Real As = S;
if (As < 0) As = - As;
Standard_ConstructionError_Raise_if
(As <= gp::Resolution(),"gp_Trsf::SetScaleFactor");
scale = S;
As = scale - 1.;
if (As < 0) As = - As;
Standard_Boolean unit = As <= gp::Resolution();
As = scale + 1.;
if (As < 0) As = - As;
Standard_Boolean munit = As <= gp::Resolution();
switch (shape) {
case gp_Identity :
case gp_Translation :
if (!unit) shape = gp_Scale;
if (munit) shape = gp_PntMirror;
break;
case gp_Rotation :
if (!unit) shape = gp_CompoundTrsf;
break;
case gp_PntMirror :
case gp_Ax1Mirror :
case gp_Ax2Mirror :
if (!munit) shape = gp_Scale;
if (unit) shape = gp_Identity;
break;
case gp_Scale :
if (unit) shape = gp_Identity;
if (munit) shape = gp_PntMirror;
break;
case gp_CompoundTrsf :
break;
case gp_Other :
break;
}
}
//=======================================================================
//function : SetValues
//purpose :
// 06-01-1998 modified by PMN : On utilise TolDist pour evaluer si les coeffs
// sont nuls : c'est toujours mieux que gp::Resolution !
//=======================================================================
void gp_Trsf::SetValues(const Standard_Real a11,
const Standard_Real a12,
const Standard_Real a13,
const Standard_Real a14,
const Standard_Real a21,
const Standard_Real a22,
const Standard_Real a23,
const Standard_Real a24,
const Standard_Real a31,
const Standard_Real a32,
const Standard_Real a33,
const Standard_Real a34,
// const Standard_Real Tolang,
const Standard_Real ,
const Standard_Real
#ifndef No_Exception
TolDist
#endif
)
{
gp_XYZ col1(a11,a21,a31);
gp_XYZ col2(a12,a22,a32);
gp_XYZ col3(a13,a23,a33);
gp_XYZ col4(a14,a24,a34);
// compute the determinant
gp_Mat M(col1,col2,col3);
Standard_Real s = M.Determinant();
Standard_Real As = s;
if (As < 0) As = - As;
Standard_ConstructionError_Raise_if
(As < gp::Resolution(),"gp_Trsf::SeValues, null determinant");
if (s > 0)
s = Pow(s,1./3.);
else
s = -Pow(-s,1./3.);
M.Divide(s);
// check if the matrix is a rotation matrix
// the transposition should be the invert.
gp_Mat TM(M);
TM.Transpose();
TM.Multiply(M);
//
// don t trust the initial values !
//
gp_Mat anIdentity ;
anIdentity.SetIdentity() ;
TM.Subtract(anIdentity);
As = TM.Value(1,1);
if (As < 0) As = - As;
Standard_ConstructionError_Raise_if
(As > TolDist,"gp_Trsf::SeValues, non uniform");
As = TM.Value(1,2);
if (As < 0) As = - As;
Standard_ConstructionError_Raise_if
(As > TolDist,"gp_Trsf::SeValues, non uniform");
As = TM.Value(1,3);
if (As < 0) As = - As;
Standard_ConstructionError_Raise_if
(As > TolDist,"gp_Trsf::SeValues, non uniform");
As = TM.Value(2,1);
if (As < 0) As = - As;
Standard_ConstructionError_Raise_if
(As > TolDist,"gp_Trsf::SeValues, non uniform");
As = TM.Value(2,2);
if (As < 0) As = - As;
Standard_ConstructionError_Raise_if
(As > TolDist,"gp_Trsf::SeValues, non uniform");
As = TM.Value(2,3);
if (As < 0) As = - As;
Standard_ConstructionError_Raise_if
(As > TolDist,"gp_Trsf::SeValues, non uniform");
As = TM.Value(3,1);
if (As < 0) As = - As;
Standard_ConstructionError_Raise_if
(As > TolDist,"gp_Trsf::SeValues, non uniform");
As = TM.Value(3,2);
if (As < 0) As = - As;
Standard_ConstructionError_Raise_if
(As > TolDist,"gp_Trsf::SeValues, non uniform");
As = TM.Value(3,3);
if (As < 0) As = - As;
Standard_ConstructionError_Raise_if
(As > TolDist,"gp_Trsf::SeValues, non uniform");
scale = s;
shape = gp_CompoundTrsf;
matrix = M;
loc = col4;
}
//=======================================================================
//function : GetRotation
//purpose :
//=======================================================================
gp_Quaternion gp_Trsf::GetRotation () const
{
return gp_Quaternion (matrix);
}
//=======================================================================
//function : VectorialPart
//purpose :
//=======================================================================
gp_Mat gp_Trsf::VectorialPart () const
{
if (scale == 1.0) return matrix;
gp_Mat M = matrix;
if (shape == gp_Scale || shape == gp_PntMirror)
M.SetDiagonal(scale*M.Value(1,1),
scale*M.Value(2,2),
scale*M.Value(3,3));
else
M.Multiply (scale);
return M;
}
//=======================================================================
//function : Invert
//purpose :
//=======================================================================
void gp_Trsf::Invert()
{
// -1
// X' = scale * R * X + T => X = (R / scale) * ( X' - T)
//
// Pour les gp_Trsf puisque le scale est extrait de la gp_Matrice R
// on a toujours determinant (R) = 1 et R-1 = R transposee.
if (shape == gp_Identity) { }
else if (shape == gp_Translation || shape == gp_PntMirror) loc.Reverse();
else if (shape == gp_Scale) {
Standard_Real As = scale;
if (As < 0) As = - As;
Standard_ConstructionError_Raise_if
(As <= gp::Resolution(),"");
scale = 1.0 / scale;
loc.Multiply (-scale);
}
else {
Standard_Real As = scale;
if (As < 0) As = - As;
Standard_ConstructionError_Raise_if
(As <= gp::Resolution(),"");
scale = 1.0 / scale;
matrix.Transpose ();
loc.Multiply (matrix);
loc.Multiply (-scale);
}
}
//=======================================================================
//function : Multiply
//purpose :
//=======================================================================
void gp_Trsf::Multiply(const gp_Trsf& T)
{
if (T.shape == gp_Identity) { }
else if (shape == gp_Identity) {
shape = T.shape;
scale = T.scale;
loc = T.loc;
matrix = T.matrix;
}
else if (shape == gp_Rotation && T.shape == gp_Rotation) {
if (T.loc.X() != 0.0 || T.loc.Y() != 0.0 || T.loc.Z() != 0.0) {
loc.Add (T.loc.Multiplied (matrix));
}
matrix.Multiply(T.matrix);
}
else if (shape == gp_Translation && T.shape == gp_Translation) {
loc.Add (T.loc);
}
else if (shape == gp_Scale && T.shape == gp_Scale) {
loc.Add (T.loc.Multiplied(scale));
scale = scale * T.scale;
}
else if (shape == gp_PntMirror && T.shape == gp_PntMirror) {
scale = 1.0;
shape = gp_Translation;
loc.Add (T.loc.Reversed());
}
else if (shape == gp_Ax1Mirror && T.shape == gp_Ax1Mirror) {
shape = gp_Rotation;
loc.Add (T.loc.Multiplied (matrix));
matrix.Multiply(T.matrix);
}
else if ((shape == gp_CompoundTrsf || shape == gp_Rotation ||
shape == gp_Ax1Mirror || shape == gp_Ax2Mirror)
&& T.shape == gp_Translation) {
gp_XYZ Tloc(T.loc);
Tloc.Multiply(matrix);
if (scale != 1.0) { Tloc.Multiply(scale); }
loc.Add (Tloc);
}
else if ((shape == gp_Scale || shape == gp_PntMirror)
&& T.shape == gp_Translation) {
gp_XYZ Tloc(T.loc);
Tloc.Multiply (scale);
loc.Add (Tloc);
}
else if (shape == gp_Translation &&
(T.shape == gp_CompoundTrsf || T.shape == gp_Rotation ||
T.shape == gp_Ax1Mirror || T.shape == gp_Ax2Mirror)) {
shape = gp_CompoundTrsf;
scale = T.scale;
loc.Add (T.loc);
matrix = T.matrix;
}
else if (shape == gp_Translation &&
(T.shape == gp_Scale || T.shape == gp_PntMirror)) {
shape = T.shape;
loc.Add (T.loc);
scale = T.scale;
}
else if ((shape == gp_PntMirror || shape == gp_Scale) &&
(T.shape == gp_PntMirror || T.shape == gp_Scale)) {
shape = gp_CompoundTrsf;
gp_XYZ Tloc(T.loc);
Tloc.Multiply (scale);
loc.Add (Tloc);
scale = scale * T.scale;
}
else if ((shape == gp_CompoundTrsf || shape == gp_Rotation ||
shape == gp_Ax1Mirror || shape == gp_Ax2Mirror)
&& (T.shape == gp_Scale || T.shape == gp_PntMirror)) {
shape = gp_CompoundTrsf;
gp_XYZ Tloc(T.loc);
if (scale == 1.0) {
scale = T.scale;
Tloc.Multiply(matrix);
}
else {
Tloc.Multiply (matrix);
Tloc.Multiply (scale);
scale = scale * T.scale;
}
loc.Add (Tloc);
}
else if ((T.shape == gp_CompoundTrsf || T.shape == gp_Rotation ||
T.shape == gp_Ax1Mirror || T.shape == gp_Ax2Mirror)
&& (shape == gp_Scale || shape == gp_PntMirror)) {
shape = gp_CompoundTrsf;
gp_XYZ Tloc(T.loc);
Tloc.Multiply(scale);
loc.Add (Tloc);
scale = scale * T.scale;
matrix = T.matrix;
}
else {
shape = gp_CompoundTrsf;
gp_XYZ Tloc(T.loc);
Tloc.Multiply (matrix);
if (scale != 1.0) {
Tloc.Multiply (scale);
scale = scale * T.scale;
}
else { scale = T.scale; }
loc.Add (Tloc);
matrix.Multiply(T.matrix);
}
}
//=======================================================================
//function : Power
//purpose :
//=======================================================================
void gp_Trsf::Power (const Standard_Integer N)
{
if (shape == gp_Identity) { }
else {
if (N == 0) {
scale = 1.0;
shape = gp_Identity;
matrix.SetIdentity();
loc = gp_XYZ (0.0, 0.0, 0.0);
}
else if (N == 1) { }
else if (N == -1) { Invert(); }
else {
if (N < 0) { Invert(); }
if (shape == gp_Translation) {
Standard_Integer Npower = N;
if (Npower < 0) Npower = - Npower;
Npower--;
gp_XYZ Temploc = loc;
while (1) {
if (IsOdd(Npower)) loc.Add (Temploc);
if (Npower == 1) break;
Temploc.Add (Temploc);
Npower = Npower/2;
}
}
else if (shape == gp_Scale) {
Standard_Integer Npower = N;
if (Npower < 0) Npower = - Npower;
Npower--;
gp_XYZ Temploc = loc;
Standard_Real Tempscale = scale;
while (1) {
if (IsOdd(Npower)) {
loc.Add (Temploc.Multiplied(scale));
scale = scale * Tempscale;
}
if (Npower == 1) break;
Temploc.Add (Temploc.Multiplied(Tempscale));
Tempscale = Tempscale * Tempscale;
Npower = Npower/2;
}
}
else if (shape == gp_Rotation) {
Standard_Integer Npower = N;
if (Npower < 0) Npower = - Npower;
Npower--;
gp_Mat Tempmatrix (matrix);
if (loc.X() == 0.0 && loc.Y() == 0.0 && loc.Z() == 0.0) {
while (1) {
if (IsOdd(Npower)) matrix.Multiply (Tempmatrix);
if (Npower == 1) break;
Tempmatrix.Multiply (Tempmatrix);
Npower = Npower/2;
}
}
else {
gp_XYZ Temploc = loc;
while (1) {
if (IsOdd(Npower)) {
loc.Add (Temploc.Multiplied (matrix));
matrix.Multiply (Tempmatrix);
}
if (Npower == 1) break;
Temploc.Add (Temploc.Multiplied (Tempmatrix));
Tempmatrix.Multiply (Tempmatrix);
Npower = Npower/2;
}
}
}
else if (shape == gp_PntMirror || shape == gp_Ax1Mirror ||
shape == gp_Ax2Mirror) {
if (IsEven (N)) {
shape = gp_Identity;
scale = 1.0;
matrix.SetIdentity ();
loc.SetX(0);
loc.SetY(0);
loc.SetZ(0);
}
}
else {
shape = gp_CompoundTrsf;
Standard_Integer Npower = N;
if (Npower < 0) Npower = - Npower;
Npower--;
gp_XYZ Temploc = loc;
Standard_Real Tempscale = scale;
gp_Mat Tempmatrix (matrix);
while (1) {
if (IsOdd(Npower)) {
loc.Add ((Temploc.Multiplied (matrix)).Multiplied (scale));
scale = scale * Tempscale;
matrix.Multiply (Tempmatrix);
}
if (Npower == 1) break;
Tempscale = Tempscale * Tempscale;
Temploc.Add ( (Temploc.Multiplied (Tempmatrix)).Multiplied
(Tempscale)
);
Tempmatrix.Multiply (Tempmatrix);
Npower = Npower/2;
}
}
}
}
}
//=======================================================================
//function : PreMultiply
//purpose :
//=======================================================================
void gp_Trsf::PreMultiply (const gp_Trsf& T)
{
if (T.shape == gp_Identity) { }
else if (shape == gp_Identity) {
shape = T.shape;
scale = T.scale;
loc = T.loc;
matrix = T.matrix;
}
else if (shape == gp_Rotation && T.shape == gp_Rotation) {
loc.Multiply (T.matrix);
loc.Add (T.loc);
matrix.PreMultiply(T.matrix);
}
else if (shape == gp_Translation && T.shape == gp_Translation) {
loc.Add (T.loc);
}
else if (shape == gp_Scale && T.shape == gp_Scale) {
loc.Multiply (T.scale);
loc.Add (T.loc);
scale = scale * T.scale;
}
else if (shape == gp_PntMirror && T.shape == gp_PntMirror) {
scale = 1.0;
shape = gp_Translation;
loc.Reverse();
loc.Add (T.loc);
}
else if (shape == gp_Ax1Mirror && T.shape == gp_Ax1Mirror) {
shape = gp_Rotation;
loc.Multiply (T.matrix);
loc.Add (T.loc);
matrix.PreMultiply(T.matrix);
}
else if ((shape == gp_CompoundTrsf || shape == gp_Rotation ||
shape == gp_Ax1Mirror || shape == gp_Ax2Mirror)
&& T.shape == gp_Translation) {
loc.Add (T.loc);
}
else if ((shape == gp_Scale || shape == gp_PntMirror)
&& T.shape == gp_Translation) {
loc.Add (T.loc);
}
else if (shape == gp_Translation &&
(T.shape == gp_CompoundTrsf || T.shape == gp_Rotation
|| T.shape == gp_Ax1Mirror || T.shape == gp_Ax2Mirror)) {
shape = gp_CompoundTrsf;
matrix = T.matrix;
if (T.scale == 1.0) loc.Multiply (T.matrix);
else {
scale = T.scale;
loc.Multiply (matrix);
loc.Multiply (scale);
}
loc.Add (T.loc);
}
else if ((T.shape == gp_Scale || T.shape == gp_PntMirror)
&& shape == gp_Translation) {
loc.Multiply (T.scale);
loc.Add (T.loc);
scale = T.scale;
shape = T.shape;
}
else if ((shape == gp_PntMirror || shape == gp_Scale) &&
(T.shape == gp_PntMirror || T.shape == gp_Scale)) {
shape = gp_CompoundTrsf;
loc.Multiply (T.scale);
loc.Add (T.loc);
scale = scale * T.scale;
}
else if ((shape == gp_CompoundTrsf || shape == gp_Rotation ||
shape == gp_Ax1Mirror || shape == gp_Ax2Mirror)
&& (T.shape == gp_Scale || T.shape == gp_PntMirror)) {
shape = gp_CompoundTrsf;
loc.Multiply (T.scale);
loc.Add (T.loc);
scale = scale * T.scale;
}
else if ((T.shape == gp_CompoundTrsf || T.shape == gp_Rotation ||
T.shape == gp_Ax1Mirror || T.shape == gp_Ax2Mirror)
&& (shape == gp_Scale || shape == gp_PntMirror)) {
shape = gp_CompoundTrsf;
matrix = T.matrix;
if (T.scale == 1.0) loc.Multiply (T.matrix);
else {
loc.Multiply (matrix);
loc.Multiply (T.scale);
scale = T.scale * scale;
}
loc.Add (T.loc);
}
else {
shape = gp_CompoundTrsf;
loc.Multiply (T.matrix);
if (T.scale != 1.0) {
loc.Multiply (T.scale); scale = scale * T.scale;
}
loc.Add (T.loc);
matrix.PreMultiply(T.matrix);
}
}
//=======================================================================
//function : GetRotation
//purpose : algorithm from A.Korn, M.Korn, "Mathematical Handbook for
// scientists and Engineers" McGraw-Hill, 1961, ch.14.10-2.
//=======================================================================
Standard_Boolean gp_Trsf::GetRotation (gp_XYZ& theAxis,
Standard_Real& theAngle) const
{
gp_Quaternion Q = GetRotation();
gp_Vec aVec;
Q.GetVectorAndAngle (aVec, theAngle);
theAxis = aVec.XYZ();
return Standard_True;
}