698 lines
22 KiB
C++
698 lines
22 KiB
C++
/***************************************************************************
|
|
* Copyright (c) 2013 Luke Parry <l.parry@warwick.ac.uk> *
|
|
* *
|
|
* This file is part of the FreeCAD CAx development system. *
|
|
* *
|
|
* This library is free software; you can redistribute it and/or *
|
|
* modify it under the terms of the GNU Library General Public *
|
|
* License as published by the Free Software Foundation; either *
|
|
* version 2 of the License, or (at your option) any later version. *
|
|
* *
|
|
* This library 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 Library General Public License for more details. *
|
|
* *
|
|
* You should have received a copy of the GNU Library General Public *
|
|
* License along with this library; see the file COPYING.LIB. If not, *
|
|
* write to the Free Software Foundation, Inc., 59 Temple Place, *
|
|
* Suite 330, Boston, MA 02111-1307, USA *
|
|
* *
|
|
***************************************************************************/
|
|
|
|
#include "PreCompiled.h"
|
|
#ifndef _PreComp_
|
|
# include <QAction>
|
|
# include <QApplication>
|
|
# include <QBitmap>
|
|
# include <QContextMenuEvent>
|
|
# include <QFileInfo>
|
|
# include <QFileDialog>
|
|
# include <QGLWidget>
|
|
# include <QGraphicsEffect>
|
|
# include <QMouseEvent>
|
|
# include <QPainter>
|
|
# include <QPaintEvent>
|
|
# include <QSvgGenerator>
|
|
#include <QScrollBar>
|
|
# include <QWheelEvent>
|
|
#include <QTemporaryFile>
|
|
#include <QDomDocument>
|
|
#include <QTextStream>
|
|
#include <QFile>
|
|
#include <QLabel>
|
|
#include <QTextCodec>
|
|
#include <cmath>
|
|
#endif
|
|
|
|
#include <App/Application.h>
|
|
#include <App/Document.h>
|
|
#include <App/Material.h>
|
|
#include <Base/Console.h>
|
|
#include <Base/Parameter.h>
|
|
#include <Base/Stream.h>
|
|
#include <Gui/FileDialog.h>
|
|
#include <Gui/Selection.h>
|
|
#include <Gui/WaitCursor.h>
|
|
#include <Gui/Command.h>
|
|
#include <Gui/Application.h>
|
|
#include <Gui/Document.h>
|
|
#include <Gui/BitmapFactory.h>
|
|
#include <Gui/MainWindow.h>
|
|
#include <Gui/View3DInventor.h>
|
|
#include <Gui/View3DInventorViewer.h>
|
|
#include <Gui/NavigationStyle.h>
|
|
|
|
#include <Mod/TechDraw/App/Geometry.h>
|
|
#include <Mod/TechDraw/App/DrawPage.h>
|
|
#include <Mod/TechDraw/App/DrawTemplate.h>
|
|
#include <Mod/TechDraw/App/DrawSVGTemplate.h>
|
|
#include <Mod/TechDraw/App/DrawParametricTemplate.h>
|
|
#include <Mod/TechDraw/App/DrawViewCollection.h>
|
|
#include <Mod/TechDraw/App/DrawViewBalloon.h>
|
|
#include <Mod/TechDraw/App/DrawViewDimension.h>
|
|
//#include <Mod/TechDraw/App/LandmarkDimension.h>
|
|
#include <Mod/TechDraw/App/DrawProjGroup.h>
|
|
#include <Mod/TechDraw/App/DrawViewPart.h>
|
|
#include <Mod/TechDraw/App/DrawViewAnnotation.h>
|
|
#include <Mod/TechDraw/App/DrawViewSymbol.h>
|
|
#include <Mod/TechDraw/App/DrawViewClip.h>
|
|
#include <Mod/TechDraw/App/DrawHatch.h>
|
|
#include <Mod/TechDraw/App/DrawViewSpreadsheet.h>
|
|
#include <Mod/TechDraw/App/DrawViewImage.h>
|
|
#include <Mod/TechDraw/App/DrawLeaderLine.h>
|
|
#include <Mod/TechDraw/App/DrawRichAnno.h>
|
|
#include <Mod/TechDraw/App/DrawWeldSymbol.h>
|
|
#include <Mod/TechDraw/App/DrawTile.h>
|
|
#include <Mod/TechDraw/App/DrawTileWeld.h>
|
|
#include <Mod/TechDraw/App/DrawUtil.h>
|
|
|
|
#include "Rez.h"
|
|
#include "PreferencesGui.h"
|
|
#include "QGIDrawingTemplate.h"
|
|
#include "QGITemplate.h"
|
|
#include "QGISVGTemplate.h"
|
|
#include "TemplateTextField.h"
|
|
#include "QGIViewCollection.h"
|
|
#include "QGIViewDimension.h"
|
|
#include "QGIViewBalloon.h"
|
|
#include "QGIProjGroup.h"
|
|
#include "QGIViewPart.h"
|
|
#include "QGIViewSection.h"
|
|
#include "QGIViewAnnotation.h"
|
|
#include "QGIViewSymbol.h"
|
|
#include "QGIViewClip.h"
|
|
#include "QGIViewSpreadsheet.h"
|
|
#include "QGIViewImage.h"
|
|
#include "QGIFace.h"
|
|
#include "QGILeaderLine.h"
|
|
#include "QGIRichAnno.h"
|
|
#include "QGIWeldSymbol.h"
|
|
#include "QGITile.h"
|
|
|
|
#include "ZVALUE.h"
|
|
#include "ViewProviderPage.h"
|
|
#include "QGSPage.h"
|
|
#include "QGVPage.h"
|
|
#include "MDIViewPage.h"
|
|
#include "QGVNavStyle.h"
|
|
#include "QGVNavStyleBlender.h"
|
|
#include "QGVNavStyleTouchpad.h"
|
|
#include "QGVNavStyleCAD.h"
|
|
#include "QGVNavStyleTinkerCAD.h"
|
|
#include "QGVNavStyleGesture.h"
|
|
#include "QGVNavStyleInventor.h"
|
|
#include "QGVNavStyleMaya.h"
|
|
#include "QGVNavStyleOCC.h"
|
|
#include "QGVNavStyleOpenSCAD.h"
|
|
#include "QGVNavStyleRevit.h"
|
|
|
|
// used SVG namespaces
|
|
#define CC_NS_URI "http://creativecommons.org/ns#"
|
|
#define DC_NS_URI "http://purl.org/dc/elements/1.1/"
|
|
#define RDF_NS_URI "http://www.w3.org/1999/02/22-rdf-syntax-ns#"
|
|
#define INKSCAPE_NS_URI "http://www.inkscape.org/namespaces/inkscape"
|
|
#define SODIPODI_NS_URI "http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
|
|
|
|
/*** pan-style cursor *******/
|
|
|
|
#define PAN_WIDTH 16
|
|
#define PAN_HEIGHT 16
|
|
#define PAN_BYTES ((PAN_WIDTH + 7) / 8) * PAN_HEIGHT
|
|
#define PAN_HOT_X 7
|
|
#define PAN_HOT_Y 7
|
|
|
|
static unsigned char pan_bitmap[PAN_BYTES] =
|
|
{
|
|
0xc0, 0x03, 0x60, 0x02, 0x20, 0x04, 0x10, 0x08,
|
|
0x68, 0x16, 0x54, 0x2a, 0x73, 0xce, 0x01, 0x80,
|
|
0x01, 0x80, 0x73, 0xce, 0x54, 0x2a, 0x68, 0x16,
|
|
0x10, 0x08, 0x20, 0x04, 0x40, 0x02, 0xc0, 0x03
|
|
};
|
|
|
|
static unsigned char pan_mask_bitmap[PAN_BYTES] =
|
|
{
|
|
0xc0,0x03,0xe0,0x03,0xe0,0x07,0xf0,0x0f,0xe8,0x17,0xdc,0x3b,0xff,0xff,0xff,
|
|
0xff,0xff,0xff,0xff,0xff,0xdc,0x3b,0xe8,0x17,0xf0,0x0f,0xe0,0x07,0xc0,0x03,
|
|
0xc0,0x03
|
|
};
|
|
/*** zoom-style cursor ******/
|
|
|
|
#define ZOOM_WIDTH 16
|
|
#define ZOOM_HEIGHT 16
|
|
#define ZOOM_BYTES ((ZOOM_WIDTH + 7) / 8) * ZOOM_HEIGHT
|
|
#define ZOOM_HOT_X 5
|
|
#define ZOOM_HOT_Y 7
|
|
|
|
static unsigned char zoom_bitmap[ZOOM_BYTES] =
|
|
{
|
|
0x00, 0x0f, 0x80, 0x1c, 0x40, 0x38, 0x20, 0x70,
|
|
0x90, 0xe4, 0xc0, 0xcc, 0xf0, 0xfc, 0x00, 0x0c,
|
|
0x00, 0x0c, 0xf0, 0xfc, 0xc0, 0xcc, 0x90, 0xe4,
|
|
0x20, 0x70, 0x40, 0x38, 0x80, 0x1c, 0x00, 0x0f
|
|
};
|
|
|
|
static unsigned char zoom_mask_bitmap[ZOOM_BYTES] =
|
|
{
|
|
0x00,0x0f,0x80,0x1f,0xc0,0x3f,0xe0,0x7f,0xf0,0xff,0xf0,0xff,0xf0,0xff,0x00,
|
|
0x0f,0x00,0x0f,0xf0,0xff,0xf0,0xff,0xf0,0xff,0xe0,0x7f,0xc0,0x3f,0x80,0x1f,
|
|
0x00,0x0f
|
|
};
|
|
using namespace Gui;
|
|
using namespace TechDraw;
|
|
using namespace TechDrawGui;
|
|
|
|
class QGVPage::Private : public ParameterGrp::ObserverType {
|
|
public:
|
|
/// handle to the viewer parameter group
|
|
ParameterGrp::handle hGrp;
|
|
QGVPage* page;
|
|
Private(QGVPage* page) : page(page) {
|
|
// attach parameter Observer
|
|
hGrp = App::GetApplication().GetParameterGroupByPath("User parameter:BaseApp/Preferences/View");
|
|
hGrp->Attach(this);
|
|
}
|
|
void init() {
|
|
page->m_atCursor = hGrp->GetBool("ZoomAtCursor", 1l);
|
|
page->m_invertZoom = hGrp->GetBool("InvertZoom", 0l);
|
|
page->m_zoomIncrement = hGrp->GetFloat("ZoomStep",0.02);
|
|
|
|
auto hTDPref = App::GetApplication().GetUserParameter()
|
|
.GetGroup("BaseApp")->GetGroup("Preferences")->GetGroup("Mod/TechDraw/General");
|
|
page->m_reversePan = hTDPref->GetInt("KbPan",1);
|
|
page->m_reverseScroll = hTDPref->GetInt("KbScroll",1);
|
|
}
|
|
/// Observer message from the ParameterGrp
|
|
void OnChange(ParameterGrp::SubjectType &rCaller,ParameterGrp::MessageType Reason) override {
|
|
const ParameterGrp& rGrp = static_cast<ParameterGrp&>(rCaller);
|
|
if (strcmp(Reason,"NavigationStyle") == 0) {
|
|
std::string model = rGrp.GetASCII("NavigationStyle",CADNavigationStyle::getClassTypeId().getName());
|
|
page->setNavigationStyle(model);
|
|
} else if (strcmp(Reason,"InvertZoom") == 0) {
|
|
page->m_invertZoom = rGrp.GetBool("InvertZoom", true);
|
|
} else if (strcmp(Reason,"ZoomStep") == 0) {
|
|
page->m_zoomIncrement = rGrp.GetFloat("ZoomStep", 0.0f);
|
|
} else if (strcmp(Reason,"ZoomAtCursor") == 0) {
|
|
page->m_atCursor = rGrp.GetBool("ZoomAtCursor", true);
|
|
if (page->m_atCursor) {
|
|
page->setResizeAnchor(QGVPage::AnchorUnderMouse);
|
|
page->setTransformationAnchor(QGVPage::AnchorUnderMouse);
|
|
} else {
|
|
page->setResizeAnchor(QGVPage::AnchorViewCenter);
|
|
page->setTransformationAnchor(QGVPage::AnchorViewCenter);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
QGVPage::QGVPage(ViewProviderPage *vp, QGSPage* s, QWidget *parent)
|
|
: QGraphicsView(parent),
|
|
m_renderer(Native),
|
|
drawBkg(true),
|
|
m_vpPage(nullptr),
|
|
m_scene(s),
|
|
balloonPlacing(false),
|
|
panningActive(false),
|
|
m_showGrid(false),
|
|
m_navStyle(nullptr),
|
|
d(new Private(this))
|
|
{
|
|
assert(vp);
|
|
m_vpPage = vp;
|
|
const char* name = vp->getDrawPage()->getNameInDocument();
|
|
setObjectName(QString::fromLocal8Bit(name));
|
|
m_vpPage->setGraphicsView(this);
|
|
|
|
setScene(s);
|
|
setMouseTracking(true);
|
|
viewport()->setMouseTracking(true);
|
|
|
|
m_parentMDI = static_cast<MDIViewPage*>(parent);
|
|
m_saveContextEvent = nullptr;
|
|
|
|
setViewportUpdateMode(QGraphicsView::FullViewportUpdate); //this prevents crash when deleting dims.
|
|
//scene(view?) indices of dirty regions gets
|
|
//out of sync. missing prepareGeometryChange
|
|
//somewhere???? QTBUG-18021????
|
|
setCacheMode(QGraphicsView::CacheBackground);
|
|
|
|
d->init();
|
|
if (m_atCursor) {
|
|
setResizeAnchor(AnchorUnderMouse);
|
|
setTransformationAnchor(AnchorUnderMouse);
|
|
} else {
|
|
setResizeAnchor(AnchorViewCenter);
|
|
setTransformationAnchor(AnchorViewCenter);
|
|
}
|
|
setAlignment(Qt::AlignCenter);
|
|
|
|
// setDragMode(ScrollHandDrag);
|
|
setDragMode(QGraphicsView::NoDrag);
|
|
resetCursor();
|
|
setRenderHints(QPainter::Antialiasing | QPainter::SmoothPixmapTransform);
|
|
|
|
bkgBrush = new QBrush(getBackgroundColor());
|
|
|
|
balloonCursor = new QLabel(this);
|
|
balloonCursor->setPixmap(prepareCursorPixmap("TechDraw_Balloon.svg", balloonHotspot = QPoint(8, 59)));
|
|
balloonCursor->hide();
|
|
|
|
resetCachedContent();
|
|
|
|
initNavigationStyle();
|
|
|
|
createStandardCursors(devicePixelRatio());
|
|
}
|
|
|
|
QGVPage::~QGVPage()
|
|
{
|
|
delete bkgBrush;
|
|
delete m_navStyle;
|
|
}
|
|
|
|
void QGVPage::initNavigationStyle()
|
|
{
|
|
std::string navParm = getNavStyleParameter();
|
|
setNavigationStyle(navParm);
|
|
}
|
|
|
|
void QGVPage::setNavigationStyle(std::string navParm)
|
|
{
|
|
// Base::Console().Message("QGVP::setNavigationStyle(%s)\n", navParm.c_str());
|
|
if (m_navStyle != nullptr) {
|
|
delete m_navStyle;
|
|
}
|
|
|
|
std::size_t foundBlender = navParm.find("Blender");
|
|
std::size_t foundCAD = navParm.find("Gui::CAD");
|
|
std::size_t foundTouchPad = navParm.find("Touchpad");
|
|
std::size_t foundInventor = navParm.find("Inventor");
|
|
std::size_t foundTinker = navParm.find("TinkerCAD");
|
|
std::size_t foundGesture = navParm.find("Gui::Gesture");
|
|
std::size_t foundMaya = navParm.find("Gui::Maya");
|
|
std::size_t foundOCC = navParm.find("OpenCascade");
|
|
std::size_t foundOpenSCAD = navParm.find("OpenSCAD");
|
|
std::size_t foundRevit = navParm.find("Revit");
|
|
|
|
if (foundBlender != std::string::npos) {
|
|
m_navStyle = static_cast<QGVNavStyle*>(new QGVNavStyleBlender(this));
|
|
} else if (foundCAD != std::string::npos) {
|
|
m_navStyle = static_cast<QGVNavStyle*>(new QGVNavStyleCAD(this));
|
|
} else if (foundTouchPad != std::string::npos) {
|
|
m_navStyle = static_cast<QGVNavStyle*>(new QGVNavStyleTouchpad(this));
|
|
} else if (foundInventor != std::string::npos) {
|
|
m_navStyle = static_cast<QGVNavStyle*>(new QGVNavStyleInventor(this));
|
|
} else if (foundTinker != std::string::npos) {
|
|
m_navStyle = static_cast<QGVNavStyle*>(new QGVNavStyleTinkerCAD(this));
|
|
} else if (foundGesture != std::string::npos) {
|
|
m_navStyle = static_cast<QGVNavStyle*>(new QGVNavStyleGesture(this));
|
|
} else if (foundMaya != std::string::npos) {
|
|
m_navStyle = static_cast<QGVNavStyle*>(new QGVNavStyleMaya(this));
|
|
} else if (foundOCC != std::string::npos) {
|
|
m_navStyle = static_cast<QGVNavStyle*>(new QGVNavStyleOCC(this));
|
|
} else if (foundOpenSCAD != std::string::npos) {
|
|
m_navStyle = static_cast<QGVNavStyle*>(new QGVNavStyleOpenSCAD(this));
|
|
} else if (foundRevit != std::string::npos) {
|
|
m_navStyle = static_cast<QGVNavStyle*>(new QGVNavStyleRevit(this));
|
|
} else {
|
|
m_navStyle = new QGVNavStyle(this);
|
|
}
|
|
}
|
|
|
|
void QGVPage::startBalloonPlacing(void)
|
|
{
|
|
balloonPlacing = true;
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5,15,0)
|
|
activateCursor(QCursor(balloonCursor->pixmap(Qt::ReturnByValue), balloonHotspot.x(), balloonHotspot.y()));
|
|
#else
|
|
activateCursor(QCursor(*balloonCursor->pixmap(), balloonHotspot.x(), balloonHotspot.y()));
|
|
#endif
|
|
}
|
|
|
|
void QGVPage::cancelBalloonPlacing(void)
|
|
{
|
|
balloonPlacing = false;
|
|
balloonCursor->hide();
|
|
resetCursor();
|
|
}
|
|
|
|
void QGVPage::drawBackground(QPainter *p, const QRectF &)
|
|
{
|
|
//Note: Background is not part of scene()
|
|
if(!drawBkg)
|
|
return;
|
|
|
|
if(!m_vpPage) {
|
|
return;
|
|
}
|
|
|
|
if (!m_vpPage->getDrawPage()) {
|
|
//Base::Console().Log("TROUBLE - QGVP::drawBackground - no Page Object!\n");
|
|
return;
|
|
}
|
|
|
|
p->save();
|
|
p->resetTransform();
|
|
|
|
|
|
p->setBrush(*bkgBrush);
|
|
p->drawRect(viewport()->rect().adjusted(-2,-2,2,2)); //just bigger than viewport to prevent artifacts
|
|
|
|
// Default to A3 landscape, though this is currently relevant
|
|
// only for opening corrupt docs, etc.
|
|
float pageWidth = 420,
|
|
pageHeight = 297;
|
|
|
|
if ( m_vpPage->getDrawPage()->hasValidTemplate() ) {
|
|
pageWidth = Rez::guiX(m_vpPage->getDrawPage()->getPageWidth());
|
|
pageHeight = Rez::guiX(m_vpPage->getDrawPage()->getPageHeight());
|
|
}
|
|
|
|
// Draw the white page
|
|
QRectF paperRect(0, -pageHeight, pageWidth, pageHeight);
|
|
QPolygon poly = mapFromScene(paperRect);
|
|
|
|
QBrush pageBrush(Qt::white);
|
|
p->setBrush(pageBrush);
|
|
|
|
p->drawRect(poly.boundingRect());
|
|
|
|
p->restore();
|
|
}
|
|
|
|
|
|
void QGVPage::setRenderer(RendererType type)
|
|
{
|
|
m_renderer = type;
|
|
|
|
if (m_renderer == OpenGL) {
|
|
#ifndef QT_NO_OPENGL
|
|
setViewport(new QGLWidget(QGLFormat(QGL::SampleBuffers)));
|
|
#endif
|
|
} else {
|
|
setViewport(new QWidget);
|
|
}
|
|
}
|
|
|
|
void QGVPage::setHighQualityAntialiasing(bool highQualityAntialiasing)
|
|
{
|
|
#ifndef QT_NO_OPENGL
|
|
setRenderHint(QPainter::Antialiasing, highQualityAntialiasing);
|
|
#else
|
|
Q_UNUSED(highQualityAntialiasing);
|
|
#endif
|
|
}
|
|
|
|
void QGVPage::paintEvent(QPaintEvent *event)
|
|
{
|
|
if (m_renderer == Image) {
|
|
if (m_image.size() != viewport()->size()) {
|
|
m_image = QImage(viewport()->size(), QImage::Format_ARGB32_Premultiplied);
|
|
}
|
|
|
|
QPainter imagePainter(&m_image);
|
|
QGraphicsView::render(&imagePainter);
|
|
imagePainter.end();
|
|
|
|
QPainter p(viewport());
|
|
p.drawImage(0, 0, m_image);
|
|
|
|
} else {
|
|
QGraphicsView::paintEvent(event);
|
|
}
|
|
}
|
|
|
|
void QGVPage::contextMenuEvent(QContextMenuEvent *event)
|
|
{
|
|
if (m_navStyle->allowContextMenu(event)) {
|
|
QGraphicsView::contextMenuEvent(event); //this eats the event. mouseReleaseEvent will not be called.
|
|
return;
|
|
}
|
|
|
|
//delete the old saved event before creating a new one to avoid memory leak
|
|
//NOTE: saving the actual event doesn't work as the event gets deleted somewhere in Qt
|
|
if (m_saveContextEvent != nullptr) {
|
|
delete m_saveContextEvent;
|
|
}
|
|
m_saveContextEvent = new QContextMenuEvent(QContextMenuEvent::Mouse,
|
|
event->pos(),
|
|
event->globalPos());
|
|
}
|
|
|
|
void QGVPage::pseudoContextEvent()
|
|
{
|
|
if (m_saveContextEvent != nullptr) {
|
|
m_parentMDI->contextMenuEvent(m_saveContextEvent);
|
|
}
|
|
}
|
|
|
|
void QGVPage::wheelEvent(QWheelEvent *event)
|
|
{
|
|
m_navStyle->handleWheelEvent(event);
|
|
event->accept();
|
|
}
|
|
|
|
void QGVPage::keyPressEvent(QKeyEvent *event)
|
|
{
|
|
m_navStyle->handleKeyPressEvent(event);
|
|
if (!event->isAccepted()) {
|
|
QGraphicsView::keyPressEvent(event);
|
|
}
|
|
}
|
|
|
|
void QGVPage::keyReleaseEvent(QKeyEvent *event)
|
|
{
|
|
m_navStyle->handleKeyReleaseEvent(event);
|
|
if (!event->isAccepted()) {
|
|
QGraphicsView::keyReleaseEvent(event);
|
|
}
|
|
}
|
|
|
|
void QGVPage::focusOutEvent(QFocusEvent *event) {
|
|
Q_UNUSED(event);
|
|
m_navStyle->handleFocusOutEvent(event);
|
|
}
|
|
|
|
void QGVPage::kbPanScroll(int xMove, int yMove)
|
|
{
|
|
if (xMove != 0) {
|
|
QScrollBar* hsb = horizontalScrollBar();
|
|
// int hRange = hsb->maximum() - hsb->minimum(); //default here is 100?
|
|
// int hDelta = xMove/hRange
|
|
int hStep = hsb->singleStep() * xMove * m_reversePan;
|
|
int hNow = hsb->value();
|
|
hsb->setValue(hNow + hStep);
|
|
}
|
|
if (yMove != 0) {
|
|
QScrollBar* vsb = verticalScrollBar();
|
|
int vStep = vsb->singleStep() * yMove * m_reverseScroll;
|
|
int vNow = vsb->value();
|
|
vsb->setValue(vNow + vStep);
|
|
}
|
|
}
|
|
|
|
void QGVPage::enterEvent(QEvent *event)
|
|
{
|
|
QGraphicsView::enterEvent(event);
|
|
m_navStyle->handleEnterEvent(event);
|
|
QGraphicsView::enterEvent(event);
|
|
}
|
|
|
|
void QGVPage::leaveEvent(QEvent * event)
|
|
{
|
|
m_navStyle->handleLeaveEvent(event);
|
|
QGraphicsView::leaveEvent(event);
|
|
}
|
|
|
|
void QGVPage::mousePressEvent(QMouseEvent *event)
|
|
{
|
|
m_navStyle->handleMousePressEvent(event);
|
|
QGraphicsView::mousePressEvent(event);
|
|
}
|
|
|
|
void QGVPage::mouseMoveEvent(QMouseEvent *event)
|
|
{
|
|
m_navStyle->handleMouseMoveEvent(event);
|
|
QGraphicsView::mouseMoveEvent(event);
|
|
}
|
|
|
|
void QGVPage::mouseReleaseEvent(QMouseEvent *event)
|
|
{
|
|
m_navStyle->handleMouseReleaseEvent(event);
|
|
QGraphicsView::mouseReleaseEvent(event);
|
|
resetCursor();
|
|
}
|
|
|
|
TechDraw::DrawPage* QGVPage::getDrawPage()
|
|
{
|
|
return m_vpPage->getDrawPage();
|
|
}
|
|
|
|
QColor QGVPage::getBackgroundColor()
|
|
{
|
|
Base::Reference<ParameterGrp> hGrp = App::GetApplication().GetUserParameter()
|
|
.GetGroup("BaseApp")->GetGroup("Preferences")->GetGroup("Mod/TechDraw/Colors");
|
|
App::Color fcColor;
|
|
fcColor.setPackedValue(hGrp->GetUnsigned("Background", 0x70707000));
|
|
return fcColor.asValue<QColor>();
|
|
}
|
|
|
|
double QGVPage::getDevicePixelRatio() const {
|
|
for (Gui::MDIView *view : m_vpPage->getDocument()->getMDIViews()) {
|
|
if (view->isDerivedFrom(Gui::View3DInventor::getClassTypeId())) {
|
|
return static_cast<Gui::View3DInventor *>(view)->getViewer()->devicePixelRatio();
|
|
}
|
|
}
|
|
|
|
return 1.0;
|
|
}
|
|
|
|
QPixmap QGVPage::prepareCursorPixmap(const char *iconName, QPoint &hotspot) {
|
|
|
|
QPointF floatHotspot(hotspot);
|
|
double pixelRatio = getDevicePixelRatio();
|
|
|
|
// Due to impossibility to query cursor size via Qt API, we stick to (32x32)*device_pixel_ratio
|
|
// as FreeCAD Wiki suggests - see https://wiki.freecadweb.org/HiDPI_support#Custom_cursor_size
|
|
double cursorSize = 32.0*pixelRatio;
|
|
|
|
QPixmap pixmap = Gui::BitmapFactory().pixmapFromSvg(iconName, QSizeF(cursorSize, cursorSize));
|
|
pixmap.setDevicePixelRatio(pixelRatio);
|
|
|
|
// The default (and here expected) SVG cursor graphics size is 64x64 pixels, thus we must adjust
|
|
// the 64x64 based hotspot position for our 32x32 based cursor pixmaps accordingly
|
|
floatHotspot *= 0.5;
|
|
|
|
#if !defined(Q_OS_WIN32) && !defined(Q_OS_MAC)
|
|
// On XCB platform, the pixmap device pixel ratio is not taken into account for cursor hot spot,
|
|
// therefore we must take care of the transformation ourselves...
|
|
// Refer to QTBUG-68571 - https://bugreports.qt.io/browse/QTBUG-68571
|
|
if (qGuiApp->platformName() == QLatin1String("xcb")) {
|
|
floatHotspot *= pixelRatio;
|
|
}
|
|
#endif
|
|
|
|
hotspot = floatHotspot.toPoint();
|
|
return pixmap;
|
|
}
|
|
|
|
void QGVPage::activateCursor(QCursor cursor) {
|
|
this->setCursor(cursor);
|
|
viewport()->setCursor(cursor);
|
|
}
|
|
|
|
void QGVPage::resetCursor() {
|
|
this->setCursor(Qt::ArrowCursor);
|
|
viewport()->setCursor(Qt::ArrowCursor);
|
|
}
|
|
|
|
void QGVPage::setPanCursor()
|
|
{
|
|
activateCursor(panCursor);
|
|
}
|
|
|
|
void QGVPage::setZoomCursor()
|
|
{
|
|
activateCursor(zoomCursor);
|
|
}
|
|
|
|
void QGVPage::drawForeground(QPainter *painter, const QRectF &rect)
|
|
{
|
|
Q_UNUSED(rect);
|
|
if (m_showGrid) {
|
|
QPen gridPen(PreferencesGui::gridQColor());
|
|
QPen savePen = painter->pen();
|
|
painter->setPen(gridPen);
|
|
painter->drawPath(m_gridPath);
|
|
painter->setPen(savePen);
|
|
}
|
|
}
|
|
|
|
void QGVPage::makeGrid(int gridWidth, int gridHeight, double gridStep)
|
|
{
|
|
QPainterPath grid;
|
|
double width = Rez::guiX(gridWidth);
|
|
double height = Rez::guiX(gridHeight);
|
|
double step = Rez::guiX(gridStep);
|
|
double horizStart = 0.0;
|
|
double vPos = 0;
|
|
int rows = (height / step) + 1;
|
|
//draw horizontal lines
|
|
for (int i = 0; i < rows; i++) {
|
|
vPos = i * step;
|
|
QPointF start (horizStart, -vPos);
|
|
QPointF end (width, -vPos);
|
|
grid.moveTo(start);
|
|
grid.lineTo(end);
|
|
}
|
|
//draw vertical lines
|
|
double vertStart = 0.0;
|
|
double hPos = 0.0;
|
|
int cols = (width / step) + 1;
|
|
for (int i = 0; i < cols; i++) {
|
|
hPos = i * step;
|
|
QPointF start(hPos, -vertStart);
|
|
QPointF end(hPos, -height);
|
|
grid.moveTo(start);
|
|
grid.lineTo(end);
|
|
}
|
|
m_gridPath = grid;
|
|
}
|
|
|
|
|
|
std::string QGVPage::getNavStyleParameter()
|
|
{
|
|
ParameterGrp::handle hGrp = App::GetApplication().GetParameterGroupByPath
|
|
("User parameter:BaseApp/Preferences/View");
|
|
std::string model = hGrp->GetASCII("NavigationStyle", NavigationStyle::getClassTypeId().getName());
|
|
return model;
|
|
}
|
|
|
|
Base::Type QGVPage::getStyleType(std::string model)
|
|
{
|
|
Base::Type type = Base::Type::fromName(model.c_str());
|
|
return type;
|
|
}
|
|
|
|
void QGVPage::createStandardCursors(double dpr)
|
|
{
|
|
(void) dpr; //avoid clang warning re unused parameter
|
|
QBitmap cursor = QBitmap::fromData(QSize(PAN_WIDTH, PAN_HEIGHT), pan_bitmap);
|
|
QBitmap mask = QBitmap::fromData(QSize(PAN_WIDTH, PAN_HEIGHT), pan_mask_bitmap);
|
|
#if defined(Q_OS_WIN32)
|
|
cursor.setDevicePixelRatio(dpr);
|
|
mask.setDevicePixelRatio(dpr);
|
|
#endif
|
|
panCursor = QCursor(cursor, mask, PAN_HOT_X, PAN_HOT_Y);
|
|
|
|
cursor = QBitmap::fromData(QSize(ZOOM_WIDTH, ZOOM_HEIGHT), zoom_bitmap);
|
|
mask = QBitmap::fromData(QSize(ZOOM_WIDTH, ZOOM_HEIGHT), zoom_mask_bitmap);
|
|
#if defined(Q_OS_WIN32)
|
|
cursor.setDevicePixelRatio(dpr);
|
|
mask.setDevicePixelRatio(dpr);
|
|
#endif
|
|
zoomCursor = QCursor(cursor, mask, ZOOM_HOT_X, ZOOM_HOT_Y);
|
|
}
|
|
|
|
#include <Mod/TechDraw/Gui/moc_QGVPage.cpp>
|