mirror of
https://github.com/ra3xdh/qucs_s
synced 2025-03-28 21:13:26 +00:00
1084 lines
32 KiB
C++
1084 lines
32 KiB
C++
/***************************************************************************
|
|
rect3ddiagram.cpp
|
|
-------------------
|
|
begin : Sat Mar 5 2005
|
|
copyright : (C) 2005 by Michael Margraf
|
|
email : michael.margraf@alumni.tu-berlin.de
|
|
***************************************************************************/
|
|
|
|
/***************************************************************************
|
|
* *
|
|
* 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. *
|
|
* *
|
|
***************************************************************************/
|
|
|
|
/*!
|
|
\class Rect3DDiagram
|
|
\brief The Rect3DDiagram class implements the 3D-Cartesian diagram
|
|
*/
|
|
|
|
#include <QFontMetrics>
|
|
|
|
#if HAVE_CONFIG_H
|
|
# include <config.h>
|
|
#endif
|
|
#include <stdlib.h>
|
|
#include <cmath>
|
|
#include <float.h>
|
|
#include <limits.h>
|
|
#if HAVE_IEEEFP_H
|
|
# include <ieeefp.h>
|
|
#endif
|
|
|
|
#include "rect3ddiagram.h"
|
|
#include "main.h"
|
|
#include "misc.h"
|
|
|
|
Rect3DDiagram::Rect3DDiagram(int _cx, int _cy) : Diagram(_cx, _cy)
|
|
{
|
|
x1 = 10; // position of label text
|
|
y1 = y3 = 7;
|
|
x2 = 200; // initial size of diagram
|
|
y2 = 200;
|
|
x3 = 207; // with some distance for right axes text
|
|
|
|
Mem = pMem = 0; // auxiliary buffer for hidden lines
|
|
|
|
Name = "Rect3D"; // BUG
|
|
// symbolic diagram painting
|
|
Lines.append(new qucs::Line(0, 0, cx, 0, QPen(Qt::black,0)));
|
|
Lines.append(new qucs::Line(0, 0, 0, cy, QPen(Qt::black,0)));
|
|
Lines.append(new qucs::Line(0, 0, cx/2, cy/2, QPen(Qt::black,0)));
|
|
}
|
|
|
|
Rect3DDiagram::~Rect3DDiagram()
|
|
{
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
// Calculates the coefficients for 3D -> 2D transformation
|
|
void Rect3DDiagram::calcCoefficients()
|
|
{
|
|
double rX = double(rotX) * pi/180.0;
|
|
double rY = double(rotY) * pi/180.0;
|
|
double rZ = double(rotZ) * pi/180.0;
|
|
|
|
cxy = sin(rZ); cxx = cos(rZ);
|
|
cxz = sin(rY); rY = cos(rY);
|
|
cyz = sin(rX); czz = cos(rX);
|
|
rX = cyz*cxz; rZ = czz*cxz;
|
|
cyx = czz * cxy + rX * cxx;
|
|
cyy = czz * cxx - rX * cxy;
|
|
czx = cyz * cxy - rZ * cxx;
|
|
czy = cyz * cxx + rZ * cxy;
|
|
cxx *= rY; cxy *= -rY; cyz *= -rY; czz *= rY;
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
double Rect3DDiagram::calcX_2D(double x, double y, double z) const
|
|
{
|
|
return (cxx * x + cxy * y + cxz * z) * scaleX;
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
double Rect3DDiagram::calcY_2D(double x, double y, double z) const
|
|
{
|
|
return (cyx * x + cyy * y + cyz * z) * scaleY;
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
double Rect3DDiagram::calcZ_2D(double x, double y, double z) const
|
|
{
|
|
return czx * x + czy * y + czz * z;
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
// Determines the position of the coordinate cross, i.e. calculates
|
|
// "scaleX", "scaleY", "xorig" and "yorig". Returns the corner with
|
|
// the largest distance from the viewer.
|
|
int Rect3DDiagram::calcCross(int *Xses, int *Yses)
|
|
{
|
|
double x3D, y3D, z3D, x2D[8], y2D[8], z2D;
|
|
double XMIN_2D, XMAX_2D, YMIN_2D, YMAX_2D, ZMIN_2D;
|
|
|
|
int z, Center = 0; // used to save minimum z (is center for axis cross)
|
|
scaleX = scaleY = 1.0; // used in "calcX_2D" and "calcY_2D"
|
|
XMIN_2D = YMIN_2D = XMAX_2D = YMAX_2D = ZMIN_2D = 0.0; // origin is zero
|
|
for(z=0; z<8; z++) { // check 2D coordinates of all 8 quadrat corners
|
|
if(z & 1) x3D = 1.0; else x3D = 0.0;
|
|
if(z & 2) y3D = 1.0; else y3D = 0.0;
|
|
if(z & 4) z3D = 1.0; else z3D = 0.0;
|
|
x2D[z] = calcX_2D(x3D, y3D, z3D);
|
|
y2D[z] = calcY_2D(x3D, y3D, z3D);
|
|
z2D = calcZ_2D(x3D, y3D, z3D);
|
|
|
|
if(x2D[z] < XMIN_2D) XMIN_2D = x2D[z];
|
|
if(x2D[z] > XMAX_2D) XMAX_2D = x2D[z];
|
|
if(y2D[z] < YMIN_2D) YMIN_2D = y2D[z];
|
|
if(y2D[z] > YMAX_2D) YMAX_2D = y2D[z];
|
|
if(z2D < ZMIN_2D) { ZMIN_2D = z2D; Center = z; }
|
|
}
|
|
|
|
scaleX = double(x2) / (XMAX_2D - XMIN_2D); // scaling 3D -> 2D transformation
|
|
scaleY = double(y2) / (YMAX_2D - YMIN_2D);
|
|
xorig = -XMIN_2D * scaleX; // position of origin
|
|
yorig = -YMIN_2D * scaleY;
|
|
|
|
for(z=0; z<8; z++) { // calculate 2D coordinates of all corners
|
|
*(Xses+z) = int(x2D[z] * scaleX + 0.5 + xorig);
|
|
*(Yses+z) = int(y2D[z] * scaleY + 0.5 + yorig);
|
|
}
|
|
return Center;
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
// Is needed for markers.
|
|
void Rect3DDiagram::calcCoordinate(const double* xD, const double* zD, const double* yD,
|
|
float *px, float *py, Axis const*) const
|
|
{
|
|
double x3D = zD[0];
|
|
double y3D = zD[1];
|
|
double z3D;
|
|
if(zAxis.log) {
|
|
z3D = sqrt(x3D*x3D + y3D*y3D);
|
|
/* if(z3D <= 0.0) clipping not yet correct implemented
|
|
z3D = -1e5; // "negative infinity"
|
|
else*/
|
|
z3D = log10(z3D / fabs(zAxis.low)) / log10(zAxis.up / zAxis.low);
|
|
}
|
|
else {
|
|
if(fabs(y3D) > 1e-250) // preserve negative values if no complex number
|
|
x3D = sqrt(x3D*x3D + y3D*y3D);
|
|
z3D = (x3D - zAxis.low) / (zAxis.up - zAxis.low);
|
|
}
|
|
|
|
x3D = *(xD++);
|
|
if(xAxis.log) {
|
|
x3D /= xAxis.low;
|
|
/* if(x3D <= 0.0) clipping not yet correct implemented
|
|
x3D = -1e5; // "negative infinity"
|
|
else*/
|
|
x3D = log10(x3D) / log10(xAxis.up / xAxis.low);
|
|
}
|
|
else
|
|
x3D = (x3D - xAxis.low) / (xAxis.up - xAxis.low);
|
|
|
|
if(yAxis.log) {
|
|
y3D = (*yD) / yAxis.low;
|
|
/* if(y3D <= 0.0) clipping not yet correct implemented
|
|
y3D = -1e5; // "negative infinity"
|
|
else*/
|
|
y3D = log10(y3D) / log10(yAxis.up / yAxis.low);
|
|
}
|
|
else
|
|
y3D = (*yD - yAxis.low) / (yAxis.up - yAxis.low);
|
|
|
|
*px = float(calcX_2D(x3D, y3D, z3D)) + xorig;
|
|
*py = float(calcY_2D(x3D, y3D, z3D)) + yorig;
|
|
|
|
if(std::isfinite(*px))
|
|
if(std::isfinite(*py))
|
|
return;
|
|
|
|
*px = float(xorig);
|
|
*py = float(yorig);
|
|
}
|
|
|
|
// --------------------------------------------------------------
|
|
void Rect3DDiagram::finishMarkerCoordinates(float& fCX, float& fCY) const
|
|
{
|
|
if(!insideDiagram(fCX, fCY)) {
|
|
fCX = fCY = 0.0;
|
|
}
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
void Rect3DDiagram::calcCoordinate3D(double x, double y, double zr, double zi,
|
|
tPoint3D *p, tPointZ *pz)
|
|
{
|
|
if(zAxis.log) {
|
|
zr = sqrt(zr*zr + zi*zi);
|
|
/* if(zr <= 0.0) clipping not yet correct implemented
|
|
zr = -1e5; // "negative infinity"
|
|
else*/
|
|
zr = log10(zr / fabs(zAxis.low)) / log10(zAxis.up / zAxis.low);
|
|
}
|
|
else {
|
|
if(fabs(zi) > 1e-250) // preserve negative values if no complex number
|
|
zr = sqrt(zr*zr + zi*zi);
|
|
zr = (zr - zAxis.low) / (zAxis.up - zAxis.low);
|
|
}
|
|
|
|
if(xAxis.log) {
|
|
x /= xAxis.low;
|
|
/* if(x <= 0.0) clipping not yet correct implemented
|
|
x = -1e5; // "negative infinity"
|
|
else*/
|
|
x = log10(x) / log10(xAxis.up / xAxis.low);
|
|
}
|
|
else
|
|
x = (x - xAxis.low) / (xAxis.up - xAxis.low);
|
|
|
|
if(yAxis.log) {
|
|
y = y / yAxis.low;
|
|
/* if(y <= 0.0) clipping not yet correct implemented
|
|
y = -1e5; // "negative infinity"
|
|
else*/
|
|
y = log10(y) / log10(yAxis.up / yAxis.low);
|
|
}
|
|
else
|
|
y = (y - yAxis.low) / (yAxis.up - yAxis.low);
|
|
|
|
p->x = int(calcX_2D(x, y, zr) + 0.5 + xorig);
|
|
p->y = int(calcY_2D(x, y, zr) + 0.5 + yorig);
|
|
p->No = pz->No = p-Mem;
|
|
p->done = 0;
|
|
pz->z = float(calcZ_2D(x, y, zr));
|
|
}
|
|
|
|
// --------------------------------------------------------------
|
|
bool Rect3DDiagram::isHidden(int x, int y, tBound *Bounds, char *zBuffer)
|
|
{
|
|
// remember the boundings of the polygon
|
|
if( (Bounds+x)->max < y ) (Bounds+x)->max = y;
|
|
if( (Bounds+x)->min > y ) (Bounds+x)->min = y;
|
|
|
|
// diagram area already used ?
|
|
return ( *(zBuffer + (y>>3) + x * ((y2+7)>>3)) & (1 << (y & 7)) ) != 0;
|
|
}
|
|
|
|
// --------------------------------------------------------------
|
|
// Enlarge memory block if necessary.
|
|
void Rect3DDiagram::enlargeMemoryBlock(tPoint3D* &MemEnd)
|
|
{
|
|
if(pMem >= MemEnd) {
|
|
int Size = MemEnd - Mem + 256;
|
|
MemEnd = Mem;
|
|
Mem = (tPoint3D*)realloc(Mem, Size*sizeof(tPoint3D));
|
|
pMem += Mem - MemEnd;
|
|
MemEnd = Mem + Size - 5;
|
|
}
|
|
}
|
|
|
|
// --------------------------------------------------------------
|
|
// Calculate all 2D points of the line between point "p" and "p+1".
|
|
// Parameters: p - pointer on 3D coordinate of line start point
|
|
// (p+1 points onto line end point)
|
|
// MemEnd - pointer where memory block ends
|
|
// Bounds - memory block for occupied polygon area
|
|
// zBuffer - memory block for occupied diagram area
|
|
void Rect3DDiagram::calcLine(tPoint3D* &p, tPoint3D* &MemEnd,
|
|
tBound *Bounds, char *zBuffer)
|
|
{
|
|
int Pos_;
|
|
int x1_ = p->x, y1_ = p->y;
|
|
int x2_ = (p+1)->x, y2_ = (p+1)->y;
|
|
|
|
bool wasHidden = isHidden(x1_, y1_, Bounds, zBuffer);
|
|
if(wasHidden)
|
|
if((p->done & 1) == 0)
|
|
p->done |= 4; // mark as hidden
|
|
|
|
int ax_ = 0, ay_ = 0;
|
|
int ix_, iy_, dx_, dy_, of_;
|
|
|
|
if(x2_ >= x1_) {
|
|
dx_ = x2_ - x1_;
|
|
ix_ = 1;
|
|
}
|
|
else {
|
|
dx_ = x1_ - x2_;
|
|
ix_ = -1;
|
|
}
|
|
|
|
if(y2_ >= y1_) {
|
|
dy_ = y2_ - y1_;
|
|
iy_ = 1;
|
|
}
|
|
else {
|
|
dy_ = y1_ - y2_;
|
|
iy_ = -1;
|
|
}
|
|
|
|
if(dx_ < dy_) {
|
|
of_ = dx_; // exchange dx and dy
|
|
dx_ = dy_;
|
|
dy_ = of_;
|
|
|
|
ax_ = iy_;
|
|
ay_ = ix_;
|
|
ix_ = iy_ = 0;
|
|
}
|
|
|
|
of_ = dx_ >> 1;
|
|
for(int i=dx_; i>1; i--) { // calculate each point of the line
|
|
x1_ += ix_;
|
|
y1_ += ax_;
|
|
of_ += dy_;
|
|
if(of_ > dx_) {
|
|
of_ -= dx_;
|
|
x1_ += ay_;
|
|
y1_ += iy_;
|
|
}
|
|
|
|
// This section has significant impact to the hiding algorithm
|
|
// be aware isHidden() modifies (Bounds+x)->min/max
|
|
// if isHidden() is replaced by false nearly all segments are drawn,
|
|
// nothing set hidden by this section ... though few segments are hidden
|
|
// if isHidden() is replaced by true most segments are not drawn
|
|
// and many line-ends are outside the diagram-are (mem/malloc-issue?)
|
|
if( isHidden(x1_, y1_, Bounds, zBuffer) != wasHidden )
|
|
if((p->done & 1) == 0) {
|
|
wasHidden = !wasHidden;
|
|
pMem->x = x1_;
|
|
pMem->y = y1_;
|
|
pMem->No = p->No;
|
|
pMem->done = 0;
|
|
if(wasHidden) pMem->done = 4; // mark as hidden
|
|
pMem++;
|
|
|
|
Pos_ = p - Mem;
|
|
// Enlarge memory block if necessary.
|
|
enlargeMemoryBlock(MemEnd); // this may make "p" invalid (realloc)
|
|
p = Mem + Pos_; // rebuild "p"
|
|
}
|
|
}
|
|
|
|
// extra treatment for last point (create no further point)
|
|
// OLD: implementation hides most of the segmets
|
|
// NEW: unhide missing segments
|
|
if(!isHidden((p+1)->x, (p+1)->y, Bounds, zBuffer))
|
|
{
|
|
if(!isHidden((p)->x, (p)->y, Bounds, zBuffer))
|
|
{
|
|
(p+1)->done &= ~4;
|
|
}
|
|
else
|
|
{
|
|
(p)->done &= ~4;
|
|
}
|
|
}
|
|
|
|
// If this assignment is commented-out all inner segments are not drawn.
|
|
// Just the surrounding boundary (all segments) of the mesh is drawn
|
|
p->done |= 1; // mark as already worked on
|
|
}
|
|
|
|
// --------------------------------------------------------------
|
|
// Compare functions for GNU qsort routine.
|
|
int Rect3DDiagram::comparePoint3D(const void *Point1, const void *Point2)
|
|
{
|
|
return ((tPoint3D*)Point1)->No - ((tPoint3D*)Point2)->No;
|
|
}
|
|
int Rect3DDiagram::comparePointZ(const void *Point1, const void *Point2)
|
|
{
|
|
if((((tPointZ*)Point2)->z - ((tPointZ*)Point1)->z) < 0.0f)
|
|
return -1;
|
|
return 1;
|
|
}
|
|
|
|
// --------------------------------------------------------------
|
|
// Removes the invisible parts of the graph.
|
|
void Rect3DDiagram::removeHiddenLines(char *zBuffer, tBound *Bounds)
|
|
{
|
|
double Dummy = 0.0; // number for 1-dimensional data in 3D cartesian
|
|
double *px, *py, *pz;
|
|
|
|
tPoint3D *p;
|
|
int i, j, z, dx, dy, Size=0;
|
|
// pre-calculate buffer size to avoid reallocations in the first step
|
|
for (Graph *g : Graphs)
|
|
if(g->cPointsY)
|
|
Size += g->axis(0)->count * g->countY;
|
|
|
|
// "Mem" should be the last malloc to simplify realloc
|
|
// multiplicator 'malloc_8xsize' added
|
|
// 'malloc_8xsize' increases the requested size in the 2 malloc lines below
|
|
// this comment section.
|
|
// And it is added to the calculation of 'MemEnd'.
|
|
// Main reason are the segments drawn to points outside the diagram.
|
|
// It is assumed to be a memory issue.
|
|
// Note: As of today the memory calculation is still not 100% clean.
|
|
int malloc_8xsize=8;
|
|
tPointZ *zMem = (tPointZ*)malloc( malloc_8xsize*(Size+2)*sizeof(tPointZ) );
|
|
Mem = (tPoint3D*)malloc( malloc_8xsize*2*(Size+2)*sizeof(tPoint3D) );
|
|
|
|
pMem = Mem;
|
|
tPointZ *zp = zMem, *zp_tmp;
|
|
|
|
// ...............................................................
|
|
for (Graph *g : Graphs) {
|
|
|
|
pz = g->cPointsY;
|
|
if(!pz) continue;
|
|
if(g->numAxes() < 1) continue;
|
|
|
|
py = &Dummy;
|
|
if(g->countY > 1) py = g->axis(1)->Points;
|
|
|
|
p = pMem; // save status for cross grid
|
|
zp_tmp = zp;
|
|
// ..........................................
|
|
// calculate coordinates of all lines
|
|
dx = g->axis(0)->count;
|
|
if(g->countY > 1) dy = g->axis(1)->count;
|
|
else dy = 0;
|
|
for(i=g->countY-1; i>=0; i--) { // y coordinates
|
|
px = g->axis(0)->Points;
|
|
|
|
for(j=dx; j>0; j--) { // x coordinates
|
|
calcCoordinate3D(*(px++), *py, *pz, *(pz+1), pMem++, zp++);
|
|
pz += 2;
|
|
}
|
|
|
|
(pMem-1)->done |= 8; // mark as "last in line"
|
|
py++;
|
|
if(dy > 0) if((i % dy) == 0)
|
|
py = g->axis(1)->Points;
|
|
}
|
|
(pMem-1)->done |= 512; // mark as "last point before grid"
|
|
|
|
// ..........................................
|
|
// copy points for cross lines ("dx", "dy" still unchanged ! )
|
|
if(g->countY > 1) {
|
|
zp = zp_tmp;
|
|
for(j=g->countY/dy; j>0; j--) { // every plane
|
|
for(i=dx; i>0; i--) { // every branch
|
|
for(z=dy; z>0; z--) { // every point
|
|
pMem->x = p->x;
|
|
pMem->y = p->y;
|
|
pMem->No = pMem-Mem;
|
|
pMem->done = 0;
|
|
zp->NoCross = pMem-Mem; // position of its cross grid
|
|
pMem++;
|
|
p += dx; // next coordinate
|
|
zp += dx;
|
|
}
|
|
(pMem-1)->done |= 8; // mark as "last in line"
|
|
p -= dx*dy - 1; // next z coordinate
|
|
zp -= dx*dy - 1;
|
|
}
|
|
p += dx*(dy-1);
|
|
zp += dx*(dy-1);
|
|
}
|
|
}
|
|
(pMem-1)->done |= 256; // mark as "very last point"
|
|
|
|
|
|
if(hideLines) {
|
|
// ..........................................
|
|
// Calculate the z-coordinate of all polygons by building the
|
|
// sum of the z-coordinates of all of its 4 corners.
|
|
// After this, each point represents one polygon. The unnecessary
|
|
// points are filled with "-FLTMAX".
|
|
zp = zp_tmp;
|
|
// "dx" and "dy" are still unchanged !
|
|
for(i=g->countY-1; i>=0; i--) { // all branches
|
|
if(dy > 0) if(i % dy) {
|
|
for(j=dx-1; j>0; j--) { // x coordinates
|
|
zp->z += (zp+1)->z + (zp+dx)->z + (zp+dx+1)->z;
|
|
zp++;
|
|
}
|
|
zp->z = -FLT_MAX; // last one not needed
|
|
zp++;
|
|
continue;
|
|
}
|
|
|
|
// last line not needed
|
|
for(j=dx; j>0; j--) { // x coordinates
|
|
zp->z = -FLT_MAX; // last one not needed
|
|
zp++;
|
|
}
|
|
}
|
|
} // of "if(hideLines)"
|
|
|
|
} // of "for(Graphs)"
|
|
|
|
|
|
if(!hideLines) { // do not hide invisible lines
|
|
free(zMem);
|
|
return;
|
|
}
|
|
|
|
#if 0
|
|
qDebug("##########################################");
|
|
qDebug("Size 1b: Size=%d, %d, %d", Size, pMem-Mem, zp-zMem);
|
|
for(tPoint3D *p=Mem; p<pMem; p++)
|
|
qDebug("xyPoints: %d/%d - %d - %d", p->x, p->y, p->No, p->done);
|
|
qDebug("------------------------------------------");
|
|
for(tPointZ *p=zMem; p-zMem<Size; p++)
|
|
qDebug("zPoints: %g - %d", p->z, p->No);
|
|
#endif
|
|
|
|
|
|
// ..........................................
|
|
// Sort z-coordinates (greatest first).
|
|
// After this the polygons that have the smallest distance to the
|
|
// viewer are on top of the list and thus, will be processed first.
|
|
qsort(zMem, Size, sizeof(tPointZ), comparePointZ);
|
|
|
|
#if 0
|
|
qDebug("--------------------------- z sorting");
|
|
for(tPointZ *p=zMem; p-zMem<Size; p++)
|
|
qDebug("zPoints: %g - %d", p->z, p->No);
|
|
#endif
|
|
|
|
|
|
// ..........................................
|
|
char *pc;
|
|
// malloc_8xsize added (see comment before malloc_8xsize declaration)
|
|
tPoint3D *MemEnd = Mem + malloc_8xsize*2*Size - 5; // limit of buffer
|
|
|
|
zp = zMem;
|
|
for (Graph *g : Graphs) {
|
|
if(!g->cPointsY) continue;
|
|
dx = g->axis(0)->count;
|
|
if(g->countY > 1) dy = g->axis(1)->count;
|
|
else dy = 1;
|
|
|
|
// look for hidden lines ...
|
|
for(int No = g->countY/dy * (dx-1)*(dy-1); No>0; No--) {
|
|
|
|
// reset the polygon bounding buffer
|
|
for(i=x2; i>=0; i--) {
|
|
(Bounds+i)->max = INT_MIN;
|
|
(Bounds+i)->min = INT_MAX;
|
|
}
|
|
|
|
// work on all 4 lines of polygon
|
|
p = Mem + zp->No; // polygon corner coordinates
|
|
calcLine(p, MemEnd, Bounds, zBuffer);
|
|
|
|
p += dx;
|
|
calcLine(p, MemEnd, Bounds, zBuffer);
|
|
|
|
p = Mem + zp->NoCross; // cross grid
|
|
calcLine(p, MemEnd, Bounds, zBuffer);
|
|
|
|
p += dy;
|
|
calcLine(p, MemEnd, Bounds, zBuffer);
|
|
|
|
// mark the area of the polygon (stored in "*Bounds") as used
|
|
for(i=x2-1; i>=0; i--) // all x coordinates
|
|
if( (Bounds+i)->max > INT_MIN) {
|
|
pc = zBuffer + i * ((y2+7)>>3);
|
|
for(j=(Bounds+i)->min; j<=(Bounds+i)->max; j++) // all y coordinates
|
|
*(pc + (j>>3)) |= (1 << (j & 7));
|
|
}
|
|
|
|
zp++; // next polygon
|
|
}
|
|
|
|
} // of "for(Graphs)"
|
|
|
|
#if 0
|
|
qDebug("--------------------------- hidden lines %d", pMem-Mem);
|
|
for(tPoint3D *p=Mem; p<pMem; p++)
|
|
qDebug("xyPoints: %d/%d - %d - %d", p->x, p->y, p->No, p->done);
|
|
#endif
|
|
|
|
free(zMem);
|
|
|
|
// sort "No" (least one first)
|
|
qsort(Mem, pMem - Mem, sizeof(tPoint3D), comparePoint3D);
|
|
|
|
#if 0
|
|
qDebug("--------------------------- last sorting %d", pMem-Mem);
|
|
for(tPoint3D *p=Mem; p<pMem; p++)
|
|
qDebug("xyPoints: %d/%d - %d - %d", p->x, p->y, p->No, p->done);
|
|
qDebug("\n");
|
|
#endif
|
|
}
|
|
|
|
// --------------------------------------------------------------
|
|
// Removes the invisible parts of the coordinate cross.
|
|
void Rect3DDiagram::removeHiddenCross(int x1_, int y1_, int x2_, int y2_,
|
|
char *zBuffer, tBound *Bounds)
|
|
{
|
|
pMem = Mem;
|
|
|
|
pMem->x = x1_;
|
|
pMem->y = y1_;
|
|
pMem->No = 0;
|
|
pMem->done = 0;
|
|
pMem++;
|
|
|
|
pMem->x = x2_;
|
|
pMem->y = y2_;
|
|
pMem->No = 1;
|
|
pMem->done = 0;
|
|
pMem++;
|
|
|
|
tPoint3D *p = Mem+6;
|
|
calcLine(Mem, p, Bounds, zBuffer);
|
|
*pMem = *(Mem+1);
|
|
*(Mem+1) = *Mem;
|
|
p = Mem+2;
|
|
do {
|
|
if(((p-1)->done & 4) == 0)
|
|
Lines.append(new qucs::Line((p-1)->x, (p-1)->y, p->x, p->y, QPen(Qt::black,0)));
|
|
p++;
|
|
} while(p <= pMem);
|
|
}
|
|
|
|
// --------------------------------------------------------------
|
|
void Rect3DDiagram::calcLimits()
|
|
{
|
|
int i;
|
|
double a, b, c;
|
|
|
|
if(xAxis.autoScale) {// check before, to preserve limit exchange (max < min)
|
|
if(xAxis.log) {
|
|
calcAxisLogScale(&xAxis, i, a, b, c, x2);
|
|
xAxis.step = 1.0;
|
|
}
|
|
else calcAxisScale(&xAxis, a, b, c, xAxis.step, double(x2));
|
|
xAxis.limit_min = xAxis.low;
|
|
xAxis.limit_max = xAxis.up;
|
|
}
|
|
|
|
if(yAxis.autoScale) {// check before, to preserve limit exchange (max < min)
|
|
if(yAxis.log) {
|
|
calcAxisLogScale(&yAxis, i, a, b, c, y2);
|
|
yAxis.step = 1.0;
|
|
}
|
|
else calcAxisScale(&yAxis, a, b, c, yAxis.step, double(y2));
|
|
yAxis.limit_min = yAxis.low;
|
|
yAxis.limit_max = yAxis.up;
|
|
}
|
|
|
|
if(zAxis.autoScale) {// check before, to preserve limit exchange (max < min)
|
|
if(zAxis.log) {
|
|
calcAxisLogScale(&zAxis, i, a, b, c, y2);
|
|
zAxis.step = 1.0;
|
|
}
|
|
else calcAxisScale(&zAxis, a, b, c, zAxis.step, double(y2));
|
|
zAxis.limit_min = zAxis.low;
|
|
zAxis.limit_max = zAxis.up;
|
|
}
|
|
}
|
|
|
|
// --------------------------------------------------------------
|
|
int Rect3DDiagram::calcAxis(Axis *Axis, int x, int y,
|
|
double xD, double phi)
|
|
{
|
|
|
|
double GridStep, corr, yD, stepD, GridNum, Expo;
|
|
double xstepD, ystepD;
|
|
|
|
QString tmp;
|
|
// get size of text using the screen-compatible metric
|
|
QFontMetrics metrics(QucsSettings.font, 0);
|
|
int maxWidth = 0;
|
|
int count, gx, gy, w;
|
|
|
|
if(phi > 0.0) Expo = phi - pi/2.0;
|
|
else Expo = phi + pi/2.0;
|
|
gx = int(5.4 * cos(Expo) + 0.5); // short grid marker lines
|
|
gy = int(5.4 * sin(Expo) + 0.5);
|
|
|
|
|
|
if(Axis->log) {
|
|
|
|
bool back = calcAxisLogScale(Axis, w, yD, stepD, corr, int(xD));
|
|
|
|
//double upD = Axis->up;
|
|
if(yD > 1.5*stepD) yD = 10.0*stepD; // always start at power of 10
|
|
if(back) {
|
|
//upD = Axis->low;
|
|
phi += pi;
|
|
xD = 0.0;
|
|
}
|
|
|
|
int xLen, yLen;
|
|
ystepD = corr * log10(yD / fabs(Axis->low));
|
|
while(ystepD <= xD) { // create all grid lines
|
|
|
|
if (engineeringNotation) tmp = misc::num2str(yD);
|
|
else tmp = misc::StringNiceNum(yD);
|
|
if(Axis->up < 0.0) tmp = '-'+tmp;
|
|
w = metrics.boundingRect(tmp).width(); // width of text
|
|
if(maxWidth < w) maxWidth = w;
|
|
|
|
xLen = int(ystepD * cos(phi) + 0.5) + x;
|
|
yLen = int(ystepD * sin(phi) + 0.5) + y;
|
|
// if(Qt::DockRight)
|
|
Texts.append(new Text(xLen+3+gx, yLen-6+gy, tmp));
|
|
// else
|
|
// Texts.append(new Text(xLen-w-2-gx, yLen-6-gy, tmp));
|
|
|
|
// short grid marks
|
|
Lines.append(new qucs::Line(xLen-gx, yLen-gy, xLen+gx, yLen+gy,
|
|
QPen(Qt::black,0)));
|
|
yD *= 10.0;
|
|
ystepD += corr;
|
|
}
|
|
|
|
}
|
|
else { // not logarithmical
|
|
calcAxisScale(Axis, GridNum, yD, stepD, GridStep, xD);
|
|
count = int((xD - yD) / stepD) + 1; // number of grids
|
|
|
|
xstepD = stepD * cos(phi);
|
|
ystepD = stepD * sin(phi);
|
|
xD = yD * cos(phi) + 0.5 + double(x);
|
|
yD = yD * sin(phi) + 0.5 + double(y);
|
|
|
|
if(Axis->up == 0.0) Expo = log10(fabs(Axis->up-Axis->low));
|
|
else Expo = log10(fabs(Axis->up));
|
|
|
|
for(; count>0; count--) {
|
|
x = int(xD);
|
|
y = int(yD);
|
|
if(fabs(GridNum) < 0.01*pow(10.0, Expo)) GridNum = 0.0; // make 0 really 0
|
|
if (engineeringNotation) tmp = misc::num2str(GridNum);
|
|
else tmp = misc::StringNiceNum(GridNum);
|
|
|
|
w = metrics.boundingRect(tmp).width(); // width of text
|
|
if(maxWidth < w) maxWidth = w;
|
|
// if(Qt::DockRight)
|
|
Texts.append(new Text(x+3+gx, y-6+gy, tmp)); // place text right
|
|
// else
|
|
// Texts.append(new Text(x-w-2-gx, y-6-gy, tmp)); // place left
|
|
GridNum += GridStep;
|
|
|
|
// short grid marks
|
|
Lines.append(new qucs::Line(x-gx, y-gy, x+gx, y+gy, QPen(Qt::black,0)));
|
|
xD += xstepD;
|
|
yD += ystepD;
|
|
}
|
|
} // of "if(Axis->log) ... else ..."
|
|
|
|
return maxWidth+5;
|
|
}
|
|
|
|
// --------------------------------------------------------------
|
|
void Rect3DDiagram::createAxis(Axis *Axis, bool Right,
|
|
int x1_, int y1_, int x2_, int y2_)
|
|
{
|
|
Q_UNUSED(Right);
|
|
|
|
DataX const *pD;
|
|
double phi, cos_phi, sin_phi;
|
|
int x, y, z, w, valid, Index = 0;
|
|
if(Axis == &yAxis) Index = 1;
|
|
|
|
QString s;
|
|
// get size of text using the screen-compatible metric
|
|
QFontMetrics metrics(QucsSettings.font, 0);
|
|
|
|
x = x2_ - x1_;
|
|
y = y2_ - y1_;
|
|
cos_phi = sqrt(double(x*x) + double(y*y));
|
|
phi = atan2(double(y), double(x));
|
|
|
|
valid = calcAxis(Axis, x1_, y1_, cos_phi, phi); // axis numbering
|
|
z = (int)cos_phi;
|
|
cos_phi = cos(phi);
|
|
sin_phi = sin(phi);
|
|
|
|
if(fabs(phi-1e-5) > pi/2.0) {
|
|
x1_ = x2_; cos_phi *= -1;
|
|
y1_ = y2_; sin_phi *= -1;
|
|
}
|
|
x = x1_ + int(double(valid)*sin_phi);
|
|
y = y1_ - int(double(valid)*cos_phi);
|
|
if(Axis->Label.isEmpty()) {
|
|
// write all labels ----------------------------------------
|
|
for (Graph *pg : Graphs) {
|
|
if(Axis != &zAxis) {
|
|
if(!pg->cPointsY) continue;
|
|
if(valid < 0) {
|
|
delete[] pg->cPointsY;
|
|
pg->cPointsY = 0;
|
|
continue;
|
|
}
|
|
pD = pg->axis(Index);
|
|
if(!pD) continue;
|
|
s = pD->Var;
|
|
}
|
|
else {
|
|
s = pg->Var;
|
|
if(!pg->cPointsY) s += INVALID_STR;
|
|
}
|
|
x += int(double(metrics.lineSpacing())*sin_phi);
|
|
y -= int(double(metrics.lineSpacing())*cos_phi);
|
|
w = metrics.boundingRect(s).width();
|
|
Texts.append(new Text(x+int(double((z-w)>>1)*cos_phi),
|
|
y+int(double((z-w)>>1)*sin_phi),
|
|
s, pg->Color, 12.0, cos_phi, sin_phi));
|
|
}
|
|
}
|
|
else {
|
|
x += int(double(metrics.lineSpacing())*sin_phi);
|
|
y -= int(double(metrics.lineSpacing())*cos_phi);
|
|
w = metrics.boundingRect(Axis->Label).width();
|
|
Texts.append(new Text(x+int(double((z-w)>>1)*cos_phi),
|
|
y+int(double((z-w)>>1)*sin_phi),
|
|
Axis->Label, Qt::black, 12.0, cos_phi, sin_phi));
|
|
}
|
|
}
|
|
|
|
// --------------------------------------------------------------
|
|
int Rect3DDiagram::calcDiagram()
|
|
{
|
|
Lines.clear();
|
|
Texts.clear();
|
|
Arcs.clear();
|
|
|
|
double GridStep, corr, zD, zDstep, GridNum;
|
|
// get size of text using the screen-compatible metric
|
|
QFontMetrics metrics(QucsSettings.font, 0);
|
|
|
|
x3 = x2 + 7;
|
|
int z, z2, o, w;
|
|
|
|
char *zBuffer=0; // hidden line algorithm
|
|
tBound *Bounds=0;
|
|
|
|
|
|
// ===== give "step" the right sign ==================================
|
|
xAxis.step = fabs(xAxis.step);
|
|
if(xAxis.limit_min > xAxis.limit_max)
|
|
xAxis.step *= -1.0;
|
|
|
|
yAxis.step = fabs(yAxis.step);
|
|
if(yAxis.limit_min > yAxis.limit_max)
|
|
yAxis.step *= -1.0;
|
|
|
|
zAxis.step = fabs(zAxis.step);
|
|
if(zAxis.limit_min > zAxis.limit_max)
|
|
zAxis.step *= -1.0;
|
|
|
|
|
|
// ===== calculate Axis.up and Axis.low ==============================
|
|
if(xAxis.log) {
|
|
if(xAxis.autoScale) {
|
|
if(xAxis.max*xAxis.min <= 0.0) goto Frame; // invalid
|
|
}
|
|
else if(xAxis.limit_min*xAxis.limit_max <= 0.0) goto Frame; // invalid
|
|
calcAxisLogScale(&xAxis, z, zD, zDstep, corr, x2);
|
|
}
|
|
else calcAxisScale(&xAxis, GridNum, zD, zDstep, GridStep, double(x2));
|
|
|
|
if(yAxis.log) {
|
|
if(yAxis.autoScale) {
|
|
if(yAxis.max*yAxis.min <= 0.0) goto Frame; // invalid
|
|
}
|
|
else if(yAxis.limit_min*yAxis.limit_max <= 0.0) goto Frame; // invalid
|
|
calcAxisLogScale(&yAxis, z, zD, zDstep, corr, x2);
|
|
}
|
|
else calcAxisScale(&yAxis, GridNum, zD, zDstep, GridStep, double(x2));
|
|
|
|
if(zAxis.log) {
|
|
if(zAxis.autoScale) {
|
|
if(zAxis.max*zAxis.min <= 0.0) goto Frame; // invalid
|
|
}
|
|
else if(zAxis.limit_min*zAxis.limit_max <= 0.0) goto Frame; // invalid
|
|
calcAxisLogScale(&zAxis, z, zD, zDstep, corr, x2);
|
|
}
|
|
else calcAxisScale(&zAxis, GridNum, zD, zDstep, GridStep, double(x2));
|
|
|
|
|
|
// === calculate transformation coefficients from rotation angles ===
|
|
calcCoefficients();
|
|
|
|
// ===== check calculate position of axes in 2D rectangle ===========
|
|
int X[8], Y[8];
|
|
o = calcCross(X, Y);
|
|
// "o" is now the index of the origin coordinates.
|
|
|
|
|
|
// ===== paint coordinate cross ====================================
|
|
// xy area
|
|
Lines.append(new qucs::Line(X[o^1], Y[o^1], X[o^3], Y[o^3], QPen(Qt::black,0)));
|
|
Lines.append(new qucs::Line(X[o^2], Y[o^2], X[o^3], Y[o^3], QPen(Qt::black,0)));
|
|
|
|
// yz area
|
|
Lines.append(new qucs::Line(X[o^2], Y[o^2], X[o^6], Y[o^6], QPen(Qt::black,0)));
|
|
Lines.append(new qucs::Line(X[o^4], Y[o^4], X[o^6], Y[o^6], QPen(Qt::black,0)));
|
|
|
|
// xz area
|
|
Lines.append(new qucs::Line(X[o^1], Y[o^1], X[o^5], Y[o^5], QPen(Qt::black,0)));
|
|
Lines.append(new qucs::Line(X[o^4], Y[o^4], X[o^5], Y[o^5], QPen(Qt::black,0)));
|
|
|
|
|
|
// ===== create axis =============================================
|
|
if(X[o^1] < X[o^2]) w = 2; // use which z axis ?
|
|
else w = 1;
|
|
|
|
z = o^2;
|
|
if(z & 1) z ^= 1; // from where to where ?
|
|
z2 = z^1;
|
|
createAxis(&xAxis, w == 2, X[z], Y[z], X[z2], Y[z2]);
|
|
|
|
z = o^1;
|
|
if(z & 2) z ^= 2; // from where to where ?
|
|
z2 = z^2;
|
|
createAxis(&yAxis, w == 1, X[z], Y[z], X[z2], Y[z2]);
|
|
|
|
z = o^w;
|
|
if(z & 4) z ^= 4; // from where to where ?
|
|
z2 = z^4;
|
|
createAxis(&zAxis, true, X[z], Y[z], X[z2], Y[z2]);
|
|
|
|
|
|
if(hideLines) {
|
|
w = (x2+1) * (y2/8 + 1);
|
|
// To store the pixel coordinates that are already used (hidden).
|
|
// Use one bit per pixel.
|
|
zBuffer = (char*)malloc(w);
|
|
memset(zBuffer, 0, w);
|
|
|
|
// To store the boundings of the current polygon.
|
|
Bounds = (tBound*)malloc((x2+1) * sizeof(tBound));
|
|
}
|
|
|
|
// hide invisible parts of graphs
|
|
removeHiddenLines(zBuffer, Bounds);
|
|
|
|
if(hideLines) {
|
|
// now hide invisible part of coordinate cross
|
|
tPoint3D *MemTmp = Mem;
|
|
Mem = (tPoint3D*)malloc( 10*sizeof(tPoint3D) );
|
|
|
|
removeHiddenCross(X[o^1], Y[o^1], X[o], Y[o], zBuffer, Bounds); // x axis
|
|
removeHiddenCross(X[o^2], Y[o^2], X[o], Y[o], zBuffer, Bounds); // y axis
|
|
removeHiddenCross(X[o^4], Y[o^4], X[o], Y[o], zBuffer, Bounds); // z axis
|
|
|
|
free(Mem);
|
|
Mem = MemTmp; // write back values
|
|
|
|
free(Bounds);
|
|
free(zBuffer);
|
|
}
|
|
else {
|
|
Lines.append(new qucs::Line(X[o], Y[o], X[o^1], Y[o^1], QPen(Qt::black,0)));
|
|
Lines.append(new qucs::Line(X[o], Y[o], X[o^2], Y[o^2], QPen(Qt::black,0)));
|
|
Lines.append(new qucs::Line(X[o], Y[o], X[o^4], Y[o^4], QPen(Qt::black,0)));
|
|
}
|
|
|
|
pMem = Mem;
|
|
return 3;
|
|
|
|
|
|
Frame: // jump here if error occurred (e.g. impossible log boundings)
|
|
Lines.append(new qucs::Line(0, y2, x2, y2, QPen(Qt::black,0)));
|
|
Lines.append(new qucs::Line(x2, y2, x2, 0, QPen(Qt::black,0)));
|
|
Lines.append(new qucs::Line(0, 0, x2, 0, QPen(Qt::black,0)));
|
|
Lines.append(new qucs::Line(0, y2, 0, 0, QPen(Qt::black,0)));
|
|
return 0;
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
// g->Points must already be empty!!!
|
|
void Rect3DDiagram::calcData(Graph *g)
|
|
{
|
|
if(!pMem) return;
|
|
if(!g->cPointsY) return;
|
|
|
|
int Size = ((2*(g->axis(0)->count) + 1) * g->countY) + 10;
|
|
Size *= 2; // memory for cross grid lines
|
|
|
|
g->resizeScrPoints(Size);
|
|
auto p = g->begin();
|
|
auto p_end = g->begin();
|
|
p_end += Size - 9; // limit of buffer
|
|
|
|
|
|
p->setStrokeEnd();
|
|
++p;
|
|
switch(g->Style) {
|
|
case GRAPHSTYLE_SOLID:
|
|
case GRAPHSTYLE_DASH:
|
|
case GRAPHSTYLE_DOT:
|
|
case GRAPHSTYLE_LONGDASH:
|
|
do {
|
|
|
|
while(1) {
|
|
if(pMem->done & 11) // is grid point ?
|
|
if(pMem->done & 4) { // is hidden
|
|
if(pMem > Mem) {
|
|
if((pMem-1)->done & 12)
|
|
break;
|
|
}
|
|
else break;
|
|
}
|
|
FIT_MEMORY_SIZE; // need to enlarge memory block ?
|
|
(p++)->setScr(pMem->x, pMem->y);
|
|
break;
|
|
}
|
|
|
|
FIT_MEMORY_SIZE; // need to enlarge memory block ?
|
|
if(pMem->done & 8) (p++)->setBranchEnd();
|
|
|
|
if(pMem->done & 4) // point invisible ?
|
|
if((p-1)->isPt())
|
|
(p++)->setStrokeEnd();
|
|
|
|
} while(((pMem++)->done & 256) == 0);
|
|
p->setGraphEnd();
|
|
break;
|
|
|
|
default: // symbol (e.g. star) at each point **********************
|
|
do {
|
|
while(1) {
|
|
if(pMem->done & 11) // is grid point ?
|
|
if(pMem->done & 4) { // is hidden
|
|
if(pMem > Mem) {
|
|
if((pMem-1)->done & 12)
|
|
break;
|
|
}
|
|
else break;
|
|
}
|
|
|
|
(p++)->setScr(pMem->x, pMem->y);
|
|
break;
|
|
}
|
|
|
|
if(pMem->done & 8)
|
|
(p++)->setBranchEnd();
|
|
} while(((pMem++)->done & 512) == 0);
|
|
p->setGraphEnd();
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
// The labels are created during "calcDiagram", but the memory
|
|
// for the coordinates is released here.
|
|
void Rect3DDiagram::createAxisLabels()
|
|
{
|
|
if(Mem) free (Mem);
|
|
Mem = 0;
|
|
pMem = 0;
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
bool Rect3DDiagram::insideDiagram(float x, float y) const
|
|
{
|
|
return (regionCode(x, y) == 0);
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
Diagram* Rect3DDiagram::newOne()
|
|
{
|
|
return new Rect3DDiagram();
|
|
}
|
|
|
|
// ------------------------------------------------------------
|
|
Element* Rect3DDiagram::info(QString& Name, char* &BitmapFile, bool getNewOne)
|
|
{
|
|
Name = QObject::tr("3D-Cartesian");
|
|
BitmapFile = (char *) "rect3d";
|
|
|
|
if(getNewOne) return new Rect3DDiagram();
|
|
return 0;
|
|
}
|
|
|
|
// vim:ts=8:sw=2:noet
|