qucs_s/qucs/imagewriter.cpp
2023-01-15 01:17:09 +03:00

412 lines
12 KiB
C++

/*
* imagewriter.cpp - implementation of writer to image
*
* Copyright (C) 2014, Yodalee, lc85301@gmail.com
*
* This file is part of Qucs
*
* Qucs 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, or (at your option)
* any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Qucs. If not, see <http://www.gnu.org/licenses/>.
*
*/
#include "schematic.h"
#include "imagewriter.h"
#include "dialogs/exportdialog.h"
#include <QtSvg>
ImageWriter::ImageWriter(QString lastfile)
{
onlyDiagram = false;
lastExportFilename = lastfile;
}
ImageWriter::~ImageWriter()
{
}
void
ImageWriter::noGuiPrint(QWidget *doc, const QString& printFile, const QString& color)
{
Schematic *sch = dynamic_cast<Schematic*>(doc);
const int bourder = 30;
int w,h,wsel,hsel,
xmin, ymin, xmin_sel, ymin_sel;
getSchWidthAndHeight(sch, w,h,xmin,ymin);
getSelAreaWidthAndHeight(sch, wsel, hsel, xmin_sel, ymin_sel);
w += bourder;
h += bourder;
wsel += bourder;
hsel += bourder;
float scal = 1.0;
if (printFile.endsWith(".svg") || printFile.endsWith(".eps")) {
QSvgGenerator* svg1 = new QSvgGenerator();
QString tempfile = printFile + ".tmp.svg";
if (!printFile.endsWith(".svg")) {
svg1->setFileName(tempfile);
} else {
svg1->setFileName(printFile);
}
svg1->setSize(QSize(1.12*w, h));
QPainter *p = new QPainter(svg1);
p->fillRect(0, 0, svg1->size().width(), svg1->size().height(), Qt::white);
ViewPainter *vp = new ViewPainter(p);
vp->init(p, 1.0, 0, 0, xmin-bourder/2, ymin-bourder/2, 1.0, 1.0);
sch->paintSchToViewpainter(vp,true,true);
delete vp;
delete p;
delete svg1;
if (!printFile.endsWith(".svg")) {
QString cmd = "inkscape";
QStringList args;
args<<"-z"<<"-D";
QString tmpf = "--file=" + tempfile;
args<<tmpf;
if (printFile.endsWith(".eps")) {
QString pf = "--export-eps=" + printFile;
args<<pf;
}
int result = QProcess::execute(cmd,args);
if (result!=0) {
QMessageBox* msg = new QMessageBox(QMessageBox::Critical,"Export to image", "Inkscape start error!", QMessageBox::Ok);
msg->exec();
delete msg;
}
QFile::remove(tempfile);
}
} else if (printFile.endsWith(".png")) {
QImage* img;
if (color == "BW") {
img = new QImage(w, h, QImage::Format_Mono);
} else {
img = new QImage(w, h, QImage::Format_RGB888);
}
QPainter* p = new QPainter(img);
p->fillRect(0, 0, w, h, Qt::white);
ViewPainter* vp = new ViewPainter(p);
vp->init(p, scal, 0, 0, xmin*scal-bourder/2, ymin*scal-bourder/2, scal,scal);
sch->paintSchToViewpainter(vp,true,true);
img->save(printFile);
delete vp;
delete p;
delete img;
} else {
fprintf(stderr, "Unsupported format of output file. \n"
"Use PNG, SVG or PDF format!\n");
return;
}
}
QString ImageWriter::getLastSavedFile()
{
return lastExportFilename;
}
// FIXME: should check if filename exists and not silently overwrite
int ImageWriter::print(QWidget *doc)
{
Schematic *sch = dynamic_cast<Schematic*>(doc);
const int border = 30;
int w,h,wsel,hsel,
xmin, ymin, xmin_sel, ymin_sel;
int status = -1;
getSchWidthAndHeight(sch, w, h, xmin, ymin);
getSelAreaWidthAndHeight(sch, wsel, hsel, xmin_sel, ymin_sel);
w += border;
h += border;
wsel += border;
hsel += border;
bool noselect = false;
if ((wsel == (border+1)) && (hsel == (border+1))) {
noselect = true;
}
ExportDialog *dlg = new ExportDialog(
w, h, wsel, hsel, lastExportFilename, noselect, 0);
if (onlyDiagram) {
dlg->setDiagram();
}
if (dlg->exec()) {
QString filename = dlg->FileToSave();
if (QFile::exists(filename)) {
int r = QMessageBox::question(0, QObject::tr("Overwrite"),
QObject::tr("File \"%1\" already exists.\nOverwrite ?")
.arg(filename),
QMessageBox::Yes|QMessageBox::No);
if (r == QMessageBox::No) {
delete dlg;
return -1;
}
}
lastExportFilename = filename;
bool exportAll;
if (dlg->isExportSelected()) {
exportAll = false;
w = wsel;
h = hsel;
xmin = xmin_sel;
ymin = ymin_sel;
} else {
exportAll = true;
}
float scal;
if (dlg->isOriginalSize()) {
scal = 1.0;
} else {
scal = (float) dlg->Xpixels()/w;
w = round(w*scal);
h = round(h*scal);
}
if (dlg->isValidFilename()) {
if (!dlg->isSvg()) {
QImage* img;
switch (dlg->getImgFormat()) {
case ExportDialog::Coloured :
img = new QImage(w,h,QImage::Format_RGB888);
break;
case ExportDialog::Monochrome :
img = new QImage(w,h,QImage::Format_Mono);
break;
default :
break;
}
QPainter* p = new QPainter(img);
p->fillRect(0, 0, w, h, Qt::white);
ViewPainter* vp = new ViewPainter(p);
vp->init(p, scal, 0, 0,
xmin*scal-border/2, ymin*scal-border/2, scal, scal);
sch->paintSchToViewpainter(vp, exportAll, true);
img->save(filename);
delete vp;
delete p;
delete img;
}
else {
QSvgGenerator* svgwriter = new QSvgGenerator();
if (dlg->needsInkscape()) {
svgwriter->setFileName(filename+".tmp.svg");
} else {
svgwriter->setFileName(filename);
}
//svgwriter->setSize(QSize(1.12*w,1.1*h));
svgwriter->setSize(QSize(1.12*w,h));
QPainter *p = new QPainter(svgwriter);
p->fillRect(0, 0, svgwriter->size().width(), svgwriter->size().height(), Qt::white);
ViewPainter *vp = new ViewPainter(p);
vp->init(p, 1.0, 0, 0, xmin-border/2, ymin-border/2, 1.0, 1.0);
sch->paintSchToViewpainter(vp,exportAll,true);
delete vp;
delete p;
delete svgwriter;
if (dlg->needsInkscape()) {
QString cmd = "inkscape";
QStringList args;
args<<"-z"<<"-D";
QString stmp = "--file=" + filename+".tmp.svg";
args<<stmp;
if (dlg->isPdf_Tex()) {
QString tmp = filename;
tmp.chop(4);
stmp = "--export-pdf="+ tmp + " --export-latex";
args<<stmp;
}
if (dlg->isPdf()) {
stmp = "--export-pdf=" + filename;
args<<stmp;
}
if (dlg->isEps()) {
stmp = "--export-eps=" + filename;
args<<stmp;
}
int result = QProcess::execute(cmd,args);
if (result!=0) {
QMessageBox::critical(0, QObject::tr("Export to image"),
QObject::tr("Inkscape start error!"), QMessageBox::Ok);
}
QFile::remove(filename+".tmp.svg");
}
}
if (QFile::exists(filename)) {
//QMessageBox::information(0, QObject::tr("Export to image"),
// QObject::tr("Successfully exported"), QMessageBox::Ok);
status = 0;
}
else {
QMessageBox::information(0, QObject::tr("Export to image"),
QObject::tr("Disk write error!"), QMessageBox::Ok);
status = -1;
}
} else {
QMessageBox::critical(0, QObject::tr("Export to image"),
QObject::tr("Unsupported format of graphics file. \n"
"Use PNG, JPEG or SVG graphics!"), QMessageBox::Ok);
status = -1;
}
}
delete dlg;
return status;
}
void ImageWriter::getSchWidthAndHeight(Schematic *sch, int &w, int &h, int &xmin, int &ymin)
{
int xmax,ymax;
sch->sizeOfAll(xmin,ymin,xmax,ymax);
w = abs(xmax - xmin);
h = abs(ymax - ymin);
int f_w, f_h;
if (sch->sizeOfFrame(f_w,f_h)) {
xmin = std::min(0,xmin); // For components
ymin = std::min(0,ymin); // that fall out of frame
w = abs(std::max(f_w,sch->UsedX2) - xmin);
h = abs(std::max(f_h,sch->UsedY2) - ymin);
}
}
void ImageWriter::getSelAreaWidthAndHeight(Schematic *sch, int &wsel, int &hsel, int& xmin_sel_, int& ymin_sel_)
{
int xmin= INT_MAX,
ymin= INT_MAX,
xmax= INT_MIN,
ymax= INT_MIN;
for(Component *pc = sch->Components->first(); pc != 0; pc = sch->Components->next()) {
if (pc->isSelected) {
int x1,y1,x2,y2;
pc->entireBounds(x1,y1,x2,y2,sch->textCorr());
updateMinMax(xmin,xmax,ymin,ymax,x1,x2,y1,y2);
}
}
for(Wire *pw = sch->Wires->first(); pw != nullptr; pw = sch->Wires->next()) {
if (pw->isSelected) {
if(pw->x1 < xmin) xmin = pw->x1;
if(pw->x2 > xmax) xmax = pw->x2;
if(pw->y1 < ymin) ymin = pw->y1;
if(pw->y2 > ymax) ymax = pw->y2;
qDebug() << pw->x1 << pw->x2 << pw->y1 << pw->y2;
}
if (pw->Label) {
WireLabel *pl = pw->Label;
if (pl->isSelected) {
int x1,y1,x2,y2;
pl->getLabelBounding(x1,y1,x2,y2);
qDebug()<<x1<<y1<<x2<<y2;
updateMinMax(xmin,xmax,ymin,ymax,x1,x2,y1,y2);
}
}
}
for(Node *pn = sch->Nodes->first(); pn != nullptr; pn = sch->Nodes->next()) {
WireLabel *pl = pn->Label;
if(pl) { // check position of node label
if (pl->isSelected) {
int x1,x2,y1,y2;
pl->getLabelBounding(x1,y1,x2,y2);
if(x1 < xmin) xmin = x1;
if(x2 > xmax) xmax = x2;
if(y1 < ymin) ymin = y1;
if(y2 > ymax) ymax = y2;
}
}
}
for(Diagram *pd = sch->Diagrams->first(); pd != nullptr; pd =sch-> Diagrams->next()) {
if (pd->isSelected) {
int x1,y1,x2,y2;
pd->Bounding(x1,y1,x2,y2);
updateMinMax(xmin,xmax,ymin,ymax,x1,x2,y1,y2);
for (auto& pg: qAsConst(pd->Graphs)) {
for (auto& pm: qAsConst(pg->Markers)) {
if (pm->isSelected) {
//int x1,y1,x2,y2;
pm->Bounding(x1,y1,x2,y2);
updateMinMax(xmin,xmax,ymin,ymax,x1,x2,y1,y2);
}
}
}
}
}
for(Painting *pp = sch->Paintings->first(); pp != nullptr; pp = sch->Paintings->next()) {
if (pp->isSelected) {
int x1,y1,x2,y2;
pp->Bounding(x1,y1,x2,y2);
updateMinMax(xmin,xmax,ymin,ymax,x1,x2,y1,y2);
}
}
wsel = abs(xmax - xmin);
hsel = abs(ymax - ymin);
xmin_sel_ = xmin;
ymin_sel_ = ymin;
}
// Compare object (component, diagram, etc) coordinates and
// current corner coordinates and update it
void ImageWriter::updateMinMax(int &xmin, int &xmax, int &ymin, int &ymax,
int x1, int x2, int y1, int y2)
{
int d1 = std::min(x1,x2);
if (d1<xmin) xmin = d1;
int d2 = std::max(x2,x1);
if (d2>xmax) xmax = d2;
int d3 = std::min(y1,y2);
if (d3<ymin) ymin = d3;
int d4 = std::max(y2,y1);
if (d4>ymax) ymax = d4;
}