/* For general Scribus (>=1.3.2) copyright and licensing information please refer to the COPYING file provided with the program. Following this notice may exist a copyright and/or license notice that predates the release of Scribus 1.3.2 for which a new license (GPL+exception) is in place. */ /*************************************************************************** * * * 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. * * * ***************************************************************************/ #if defined(_MSC_VER) #define _USE_MATH_DEFINES #endif #include // #include #include #include "canvas.h" #include "canvasmode.h" #include "hruler.h" #include "page.h" #include "pageitem_textframe.h" #include "prefsmanager.h" #include "scribusdoc.h" #include "scpainter.h" #include "scribusview.h" #include "selection.h" #include "util.h" #include "units.h" #include "vruler.h" #define DRAW_DEBUG_LINES 0 static QPoint contentsToViewport(QPoint p) { return p; } void CanvasViewMode::init() { scale = 1; previewMode = false; viewAsPreview = false; previewVisual = -1; m_MouseButtonPressed = false; operItemMoving = false; operItemResizing = false; operItemSelecting = false; redrawPolygon.resize(0); linkedFramesToShow.clear(); drawSelectedItemsWithControls = false; drawFramelinksWithContents = false; forceRedraw = false; } QDataStream &operator<< ( QDataStream & ds, const CanvasViewMode & vm ) { ds << vm.scale << vm.previewMode << vm.viewAsPreview << vm.previewVisual << vm.m_MouseButtonPressed << vm.operItemMoving << vm.operItemResizing << vm.operItemSelecting << vm.redrawPolygon << vm.linkedFramesToShow << vm.drawSelectedItemsWithControls << vm.drawFramelinksWithContents << vm.forceRedraw; return ds; } QDataStream &operator>> ( QDataStream & ds, CanvasViewMode & vm ) { ds >> vm.scale >> vm.previewMode >> vm.viewAsPreview >> vm.previewVisual >> vm.m_MouseButtonPressed >> vm.operItemMoving >> vm.operItemResizing >> vm.operItemSelecting >> vm.redrawPolygon >> vm.linkedFramesToShow >> vm.drawSelectedItemsWithControls >> vm.drawFramelinksWithContents >> vm.forceRedraw; return ds; } Canvas::Canvas(ScribusDoc* doc, ScribusView* parent) : QWidget(parent), m_doc(doc), m_view(parent) { setAutoFillBackground(true); setAttribute(Qt::WA_OpaquePaintEvent, true); setAttribute(Qt::WA_NoSystemBackground, true); m_buffer = /*QImage()*/QPixmap(); m_bufferRect = QRect(); m_viewMode.init(); m_renderMode = RENDER_NORMAL; m_oldMinCanvasCoordinate = canvasToLocal(QPointF(0.0, 0.0)); } // ______________________________ // Coordinate Conversion Routines: FPoint Canvas::localToCanvas(QPoint p) const { /* Allow the user to select the exact coordinate represented by a ruler mark when the mouse is lined up with the ruler, rather than returning the coordinate represented by the mathematical centre of the pixel which may not be exactly the same as the coordinate represented by the ruler. */ // (xmin, xmax) = canvas top-left double xmin = m_doc->minCanvasCoordinate.x(); double ymin = m_doc->minCanvasCoordinate.y(); // (xoff, yoff) = ruler origin relative to canvas top-left double xoff = m_doc->rulerXoffset - xmin; double yoff = m_doc->rulerYoffset - ymin; if (m_doc->guidesSettings.rulerMode) { xoff += m_doc->currentPage()->xOffset(); yoff += m_doc->currentPage()->yOffset(); } // (xsp, ysp) = spacing of ruler divisions double xsp = m_doc->view()->horizRuler->ruleSpacing(); double ysp = m_doc->view()->vertRuler->ruleSpacing(); double sc = m_viewMode.scale; // number of ruler divisions from ruler origin to ruler mark closest to the // selected mouse coordinate double xn = qRound((p.x()/sc - xoff)/xsp); double yn = qRound((p.y()/sc - yoff)/ysp); // xn*xsp + xoff, yn*ysp + yoff = distance in canvas coordinates // from canvas top-left to ruler mark closest to selected mouse coordinate // If these round to the selected mouse coordinate, use these to define // (x,y), the coordinates of the desired point relative to the canvas top-left. // Otherwise, simply scale the mouse coordinates. double x = ( qRound(sc*(xn*xsp + xoff)) == p.x() ? xn*xsp + xoff : p.x()/sc ); double y = ( qRound(sc*(yn*ysp + yoff)) == p.y() ? yn*ysp + yoff : p.y()/sc ); // Finally, add xmin,ymin to get the absolute canvas coordinates of the // desired point. return FPoint(x + xmin, y + ymin); } /* FPoint Canvas::localToCanvas(QPointF p) const { return FPoint(p.x() / m_viewMode.scale + m_doc->minCanvasCoordinate.x() , p.y() / m_viewMode.scale + m_doc->minCanvasCoordinate.y()); } */ QPoint Canvas::canvasToLocal(FPoint p) const { return QPoint(qRound((p.x() - m_doc->minCanvasCoordinate.x()) * m_viewMode.scale), qRound((p.y() - m_doc->minCanvasCoordinate.y()) * m_viewMode.scale)); } QPoint Canvas::canvasToLocal(QPointF p) const { return QPoint(qRound((p.x() - m_doc->minCanvasCoordinate.x()) * m_viewMode.scale), qRound((p.y() - m_doc->minCanvasCoordinate.y()) * m_viewMode.scale)); } QRect Canvas::canvasToLocal(QRectF p) const { return QRect(qRound((p.x() - m_doc->minCanvasCoordinate.x()) * m_viewMode.scale), qRound((p.y() - m_doc->minCanvasCoordinate.y()) * m_viewMode.scale), qRound(p.width() * m_viewMode.scale), qRound(p.height() * m_viewMode.scale)); } QPoint Canvas::canvasToGlobal(FPoint p) const { return mapToParent(QPoint(0,0)) + parentWidget()->mapToGlobal(QPoint(0, 0)) + canvasToLocal(p); } QPoint Canvas::canvasToGlobal(QPointF p) const { return mapToParent(QPoint(0,0)) + parentWidget()->mapToGlobal(QPoint(0, 0)) + canvasToLocal(p); } QRect Canvas::canvasToGlobal(QRectF p) const { return QRect(mapToParent(QPoint(0,0) + canvasToLocal(p.topLeft())) + parentWidget()->mapToGlobal(QPoint(0, 0)), QSize(qRound(p.width() * m_viewMode.scale), qRound(p.height() * m_viewMode.scale))); } FPoint Canvas::globalToCanvas(QPoint p) const { return localToCanvas(p - (mapToParent(QPoint(0, 0)) + parentWidget()->mapToGlobal(QPoint(0, 0)))); } /* FPoint Canvas::globalToCanvas(QPointF p) const { return localToCanvas(p - mapToGlobal(QPoint(0, 0))); } */ QRectF Canvas::globalToCanvas(QRect p) const { FPoint org = globalToCanvas(p.topLeft()); return QRectF(org.x(), org.y(), p.width() / m_viewMode.scale, p.height() / m_viewMode.scale); } /* QRectF Canvas::globalToCanvas(QRectF p) const { FPoint org = globalToCanvas(p.topLeft()); return QRectF(org.x(), org.y(), p.width() / m_viewMode.scale, p.height() / m_viewMode.scale); } */ // ________________________ // Tests for Finding Things: bool Canvas::hitsCanvasPoint(QPoint globalPoint, FPoint canvasPoint) const { QPoint localPoint1 = globalPoint - (mapToParent(QPoint(0,0)) + parentWidget()->mapToGlobal(QPoint(0, 0))); QPoint localPoint2 = canvasToLocal(canvasPoint); int radius = m_doc->guidesSettings.grabRad; return qAbs(localPoint1.x() - localPoint2.x()) < radius && qAbs(localPoint1.y() - localPoint2.y()) < radius; } bool Canvas::hitsCanvasPoint(QPoint globalPoint, QPointF canvasPoint) const { QPoint localPoint1 = globalPoint - (mapToParent(QPoint(0,0)) + parentWidget()->mapToGlobal(QPoint(0, 0))); QPoint localPoint2 = canvasToLocal(canvasPoint); int radius = m_doc->guidesSettings.grabRad; return qAbs(localPoint1.x() - localPoint2.x()) < radius && qAbs(localPoint1.y() - localPoint2.y()) < radius; } QRect Canvas::exposedRect() const { int ex ( -(x() / m_viewMode.scale) + m_doc->minCanvasCoordinate.x() ); int ey ( -(y() / m_viewMode.scale) + m_doc->minCanvasCoordinate.y() ); int ew ( (m_view->visibleWidth() * 1.2) / m_viewMode.scale ); int eh ( (m_view->visibleHeight() * 1.2) / m_viewMode.scale ); return QRect( ex, ey, ew, eh ); } /*! returns -1 if canvasPoint is outside the frame + grabradius. returns frameHandle if canvasPoint is near a framehandle otherwise 0 */ Canvas::FrameHandle Canvas::frameHitTest(QPointF canvasPoint, PageItem* item) const { // As item->getTransform() will translate to Pos, we need to adjust the rect passed // to frameHitTest in order to take line width, which in very *creative* design can // be huge, into account. // ### might be interesting to investigate if it would be painless to just change // PageItem::getTransform. double extraS = 0.0; extraS = (item->visualHeight() - item->height()) / - 2.0; // if (item->lineColor() != CommonStrings::None) // extraS = (item->lineWidth() / -2.0); Canvas::FrameHandle result = frameHitTest(item->getTransform().inverted().map(canvasPoint), QRectF(extraS, extraS, item->visualWidth(), item->visualHeight())); // qDebug() << "frameHitTest for item" << item->ItemNr // << item->getTransform().inverted().map(canvasPoint) // << item->getTransform().inverted() // << QRectF(0, 0, item->width(), item->height()); return result; } /// Little helper to calculate |p|^2 static double length2(const QPointF& p) { return p.x()*p.x() + p.y()*p.y(); } Canvas::FrameHandle Canvas::frameHitTest(QPointF canvasPoint, QRectF frame) const { FrameHandle result = INSIDE; const double radius = m_doc->guidesSettings.grabRad / m_viewMode.scale; const double radius2 = radius * radius; double resultDistance = radius2 * 10.0; // far off const double frameWidth = frame.width(); const double frameHeight = frame.height(); const QPointF frameOrigin = frame.topLeft(); if (canvasPoint.x() < frameOrigin.x() - radius || canvasPoint.x() > frameOrigin.x() + frameWidth + radius || canvasPoint.y() < frameOrigin.y() - radius || canvasPoint.y() > frameOrigin.y() + frameHeight + radius) { // qDebug() << "framehittest" << canvasPoint << frame << "-> OUTSIDE"; return OUTSIDE; } QPointF framePoint = frameOrigin; double distance = length2(canvasPoint - framePoint); if (distance < radius2 && distance < resultDistance) { result = NORTHWEST; resultDistance = distance; } framePoint.setX(frameOrigin.x() + frameWidth/2); distance = length2(canvasPoint - framePoint); if (distance < radius2 && distance < resultDistance) { result = NORTH; resultDistance = distance; } framePoint.setX(frameOrigin.x() + frameWidth); distance = length2(canvasPoint - framePoint); if (distance < radius2 && distance < resultDistance) { result = NORTHEAST; resultDistance = distance; } framePoint.setY(frameOrigin.y() + frameHeight/2); distance = length2(canvasPoint - framePoint); if (distance < radius2 && distance < resultDistance) { result = EAST; resultDistance = distance; } framePoint.setY(frameOrigin.y() + frameHeight); distance = length2(canvasPoint - framePoint); if (distance < radius2 && distance < resultDistance) { result = SOUTHEAST; resultDistance = distance; } framePoint.setX(frameOrigin.x() + frameWidth/2); distance = length2(canvasPoint - framePoint); if (distance < radius2 && distance < resultDistance) { result = SOUTH; resultDistance = distance; } framePoint.setX(frameOrigin.x()); distance = length2(canvasPoint - framePoint); if (distance < radius2 && distance < resultDistance) { result = SOUTHWEST; resultDistance = distance; } framePoint.setY(frameOrigin.y() + frameHeight/2); distance = length2(canvasPoint - framePoint); if (distance < radius2 && distance < resultDistance) { result = WEST; //resultDistance = distance; } // qDebug() << "framehittest" << canvasPoint << frame << "->" << result; return result; } PageItem* Canvas::itemUnderCursor(QPoint globalPos, PageItem* itemAbove, bool allowInGroup, bool allowMasterItems) const { PageItem* currItem; QRectF mouseArea = globalToCanvas(QRect(globalPos, QSize(2*m_doc->guidesSettings.grabRad, 2*m_doc->guidesSettings.grabRad))); // look for masterpage items first if (allowMasterItems && !m_doc->masterPageMode() && m_doc->currentPage()->FromMaster.count() != 0) { Page* Mp = m_doc->MasterPages.at(m_doc->MasterNames[m_doc->currentPage()->MPageNam]); // if itemAbove is given, we expect to find it among the masterpage items of this page int currNr = itemAbove? m_doc->currentPage()->FromMaster.indexOf(itemAbove)-1 : m_doc->currentPage()->FromMaster.count()-1; if (currNr < 0) return NULL; while (currNr >= 0) { currItem = m_doc->currentPage()->FromMaster.at(currNr); QMatrix itemPos; if ((currItem->LayerNr == m_doc->activeLayer()) && (!m_doc->layerLocked(currItem->LayerNr))) { if (!currItem->ChangedMasterItem) { itemPos.translate(-Mp->xOffset() + m_doc->currentPage()->xOffset(), -Mp->yOffset() + m_doc->currentPage()->yOffset()); } currItem->getTransform(itemPos); QPainterPath currPath(itemPos.map(QPointF(0,0))); currPath.lineTo(itemPos.map(QPointF(currItem->width(), 0))); currPath.lineTo(itemPos.map(QPointF(currItem->width(), currItem->height()))); currPath.lineTo(itemPos.map(QPointF(0, currItem->height()))); currPath.closeSubpath(); QPainterPath currClip; currClip.addPolygon(itemPos.map(QPolygonF(currItem->Clip))); currClip.closeSubpath(); // qDebug() << "itemUnderCursor: master" << currNr << mouseArea << currPath.toFillPolygon() << currPath.intersects(mouseArea); // if (currClip.intersects(mouseArea)) // qDebug() << " within Clip:" << currClip.toFillPolygon(); if (currPath.intersects(mouseArea) || currClip.intersects(mouseArea)) // if ((QRegion(itemPos.map(QPolygon(QRect(0, 0, static_cast(currItem->width()), static_cast(currItem->height()))))).contains(mpo.toRect())) || // (QRegion(currItem->Clip * itemPos).contains(mpo.toRect()))) { if (currItem->Groups.count() != 0 && !allowInGroup) { // look for the group control // FIXME: Items or MasterPageItems?? for (int ga=0; ga < m_doc->Items->count(); ++ga) { if (m_doc->Items->at(ga)->Groups.count() != 0 && m_doc->Items->at(ga)->Groups.top() == currItem->Groups.top() && m_doc->Items->at(ga)->controlsGroup()) { currItem = m_doc->Items->at(ga); break; } } } return currItem; } } --currNr; } } // now look for normal items if (m_doc->Items->count() == 0) return NULL; int currNr = itemAbove? m_doc->Items->indexOf(itemAbove)-1 : m_doc->Items->count()-1; // qDebug() << "itemUnderCursor searching from" << currNr << itemAbove << m_doc->Items->count(); while (currNr >= 0) { currItem = m_doc->Items->at(currNr); if ((m_doc->masterPageMode()) && (!((currItem->OwnPage == -1) || (currItem->OwnPage == static_cast(m_doc->currentPage()->pageNr()))))) { --currNr; continue; } if ((currItem->LayerNr == m_doc->activeLayer()) && (!m_doc->layerLocked(currItem->LayerNr))) { QMatrix itemPos = currItem->getTransform(); QPainterPath currPath(itemPos.map(QPointF(0,0))); currPath.lineTo(itemPos.map(QPointF(currItem->width(), 0))); currPath.lineTo(itemPos.map(QPointF(currItem->width(), currItem->height()))); currPath.lineTo(itemPos.map(QPointF(0, currItem->height()))); currPath.closeSubpath(); QPainterPath currClip; currClip.addPolygon(itemPos.map(QPolygonF(currItem->Clip))); currClip.closeSubpath(); // qDebug() << "itemUnderCursor:" << currNr << mouseArea << currPath.toFillPolygon() << currPath.intersects(mouseArea); // if (currClip.intersects(mouseArea)) // qDebug() << " within Clip:" << currClip.toFillPolygon(); if (currPath.intersects(mouseArea) || currClip.intersects(mouseArea)) // if ((QRegion(itemPos.map(QPolygon(QRect(0, 0, static_cast(currItem->width()), static_cast(currItem->height()))))).contains(mpo.toRect())) || // (QRegion(currItem->Clip * itemPos).contains(mpo.toRect()))) { if (currItem->Groups.count() != 0 && !allowInGroup) { // look for the group control for (int ga=0; ga < m_doc->Items->count(); ++ga) { if (m_doc->Items->at(ga)->Groups.count() != 0 && m_doc->Items->at(ga)->Groups.top() == currItem->Groups.top() && m_doc->Items->at(ga)->controlsGroup()) { currItem = m_doc->Items->at(ga); break; } } } return currItem; } } --currNr; } return NULL; } PageItem * Canvas::itemUnderItem(PageItem * item, int& index) const { int indice = qMin(index, m_doc->Items->count()); if (index < 0 || indice < 0) return NULL; QRectF baseRect(item->getBoundingRect()); int itemLevel = m_doc->layerLevelFromNumber(item->LayerNr); if (itemLevel < 0) return NULL; for(index = indice - 1; index >= 0; --index) { PageItem* item1 = m_doc->Items->at(index); int level = m_doc->layerLevelFromNumber(item1->LayerNr); if ((item != item1) && (level >= 0) && (level <= itemLevel)) { if ((level == itemLevel) && (item1->ItemNr > item->ItemNr)) continue; QRectF uRect(item1->getBoundingRect()); if(baseRect.intersects(uRect)) return item1; } } return NULL; } // __________________ // Buffered rendering: /* Rendermodes: m_buffer holds the current page(s) m_bufferRect describes the contents in local coordinates: minCanvasCoordinate |-> local (0,0) (0,0) |-> local (scale*minCanvasCoordinate) local m_bufferRect.topLeft |-> buffer (0,0) */ void Canvas::setRenderMode(RenderMode mode) { // qDebug() << "setRenderMode" << m_renderMode << "-->" << mode; if ( (mode < RENDER_SELECTION_SEPARATE) != (m_renderMode < RENDER_SELECTION_SEPARATE) ) { clearBuffers(); } m_renderMode = mode; } void Canvas::clearBuffers() { m_buffer = /*QImage()*/QPixmap(); m_bufferRect = QRect(); m_selectionBuffer = QPixmap(); m_selectionRect = QRect(); } bool Canvas::adjustBuffer() { bool ret = false; QRect viewport(-x(), -y(), m_view->viewport()->width(), m_view->viewport()->height()); // qDebug()<<"Canvas::adjustBuffer"< viewport.left()) newRect.translate(viewport.left() - m_bufferRect.left(), 0); if (m_bufferRect.right() < viewport.right()) newRect.translate(viewport.right() - m_bufferRect.right(), 0); if (m_bufferRect.top() > viewport.top()) newRect.translate(0, viewport.top() - m_bufferRect.top()); if (m_bufferRect.bottom() < viewport.bottom()) newRect.translate(0, viewport.bottom() - m_bufferRect.bottom()); // qDebug() << "adjust buffer: " << m_bufferRect << "outside viewport" << viewport << " new rect:" << newRect; /* // enlarge buffer by half a screenwidth: QRect newRect(m_bufferRect); if (m_bufferRect.left() > viewport.left()) newRect.setLeft(qMin(newRect.left() - viewport.width()/2, viewport.left())); if (m_bufferRect.right() < viewport.right()) newRect.setRight(qMax(newRect.right() + viewport.width()/2, viewport.right())); if (m_bufferRect.top() > viewport.top()) newRect.setTop(qMin(newRect.top() - viewport.height()/2, viewport.top())); if (m_bufferRect.bottom() < viewport.bottom()) newRect.setBottom(qMax(newRect.bottom() + viewport.height()/2, viewport.bottom())); // if too large, try something smaller: if (newRect.width() > 2*viewport.width()) { newRect.setLeft(qMax(viewport.left() - viewport.width()/2, newRect.left())); newRect.setRight(qMin(viewport.right() + viewport.width()/2, newRect.right())); } if (newRect.height() > 2*viewport.height()) { newRect.setTop(qMax(viewport.top() - viewport.height()/2, newRect.top())); newRect.setBottom(qMin(viewport.bottom() + viewport.height()/2, newRect.bottom())); } */ if (!m_bufferRect.intersects(newRect)) { // qDebug() << "adjust buffer: fresh buffer" << m_bufferRect << "-->" << newRect; m_bufferRect = newRect; // m_buffer = QImage(m_bufferRect.width(), m_bufferRect.height(), QImage::Format_ARGB32); m_buffer = QPixmap(m_bufferRect.width(), m_bufferRect.height()); fillBuffer(&m_buffer, m_bufferRect.topLeft(), m_bufferRect); ret = true; #if DRAW_DEBUG_LINES QPainter p(&m_buffer); p.setPen(Qt::blue); p.drawLine(0, 0, m_buffer.width(), m_buffer.height()); p.drawLine(m_buffer.width(), 0, 0, m_buffer.height()); p.end(); #endif } else { // copy buffer: // QImage newBuffer(newRect.width(), newRect.height(), QImage::Format_ARGB32); QPixmap newBuffer(newRect.width(), newRect.height()); QPainter p(&newBuffer); int xpos = m_bufferRect.x() - newRect.x(); int ypos = m_bufferRect.y() - newRect.y(); int x = 0; int y = 0; int width = m_bufferRect.width(); int height = m_bufferRect.height(); if (xpos < 0) { x = -xpos; width -= x; xpos = 0; } if (ypos < 0) { y = -ypos; height -= y; ypos = 0; } if (xpos + width > newRect.width()) { width = newRect.width() - xpos; } if (ypos + height > newRect.height()) { height = newRect.height() - ypos; } // p.drawImage(xpos, ypos, m_buffer, x, y, width + 1, height + 1); // FIXME: == params drawPixmap? p.drawPixmap(xpos, ypos, m_buffer, x, y, width + 1, height + 1); #if DRAW_DEBUG_LINES p.setPen(Qt::blue); p.drawLine(xpos, ypos+height/2, xpos+width/2, ypos); p.drawLine(xpos+width, ypos+height/2, xpos+width/2, ypos); p.drawLine(xpos, ypos+height/2, xpos+width/2, ypos+height); p.drawLine(xpos+width, ypos+height/2, xpos+width/2, ypos+height); // qDebug() << "adjust buffer old" << m_bufferRect << "@" << xpos << ypos << "--> new" << newRect; #endif p.end(); if (newRect.top() < m_bufferRect.top()) { fillBuffer(&newBuffer, newRect.topLeft(), QRect(newRect.left(), newRect.top(), newRect.width(), m_bufferRect.top() - newRect.top() + 2)); ret = true; } if (newRect.bottom() > m_bufferRect.bottom()) { fillBuffer(&newBuffer, newRect.topLeft(), QRect(newRect.left(), m_bufferRect.bottom() - 1, newRect.width(), newRect.bottom() - m_bufferRect.bottom() + 2)); ret = true; } if (newRect.left() < m_bufferRect.left()) { fillBuffer(&newBuffer, newRect.topLeft(), QRect(newRect.left(), m_bufferRect.top(), m_bufferRect.left() - newRect.left() + 2, m_bufferRect.height())); ret = true; } if (newRect.right() > m_bufferRect.right()) { fillBuffer(&newBuffer, newRect.topLeft(), QRect(m_bufferRect.right() - 1, m_bufferRect.top(), newRect.right() - m_bufferRect.right() + 2, m_bufferRect.height())); ret = true; } m_buffer = newBuffer; m_bufferRect = newRect; #if DRAW_DEBUG_LINES QPainter p2(&m_buffer); p2.setPen(Qt::blue); p2.drawRect(xpos, ypos, width, height); p2.end(); #endif } } // else // qDebug() << "adjustBuffer: reusing" << m_bufferRect; // qDebug() << "Canvas::adjustBuffer"<rect()<isLoading()) return; // #define SHOW_ME_WHAT_YOU_GET_IN_D_CANVA #ifdef SHOW_ME_WHAT_YOU_GET_IN_D_CANVA QString dmode(""); int t1,t2,t3,t4,t5,t6; t1 = t2=t3=t4=t5 =t6= 0; t.start(); #endif // fill buffer if necessary bool bufferFilled = adjustBuffer(); // It is ugly, but until we figure out why drawing directly on the // widget is so slow, it saves us a Cray! - pm QPixmap tmpImg(p->rect().size()); QPainter qp(&tmpImg); qp.translate(-p->rect().x(), -p->rect().y()); switch (m_renderMode) { case RENDER_NORMAL: { #if DRAW_DEBUG_LINES // qDebug() << "update Buffer:" << m_bufferRect << p->rect() << m_viewMode.forceRedraw; #endif #ifdef SHOW_ME_WHAT_YOU_GET_IN_D_CANVA dmode = "NORMAL"; t1 = t.elapsed(); t.start(); #endif if ((m_viewMode.forceRedraw || m_viewMode.operTextSelecting) && (!bufferFilled)) { // qDebug() << "Canvas::paintEvent: forceRedraw=" << m_viewMode.forceRedraw << "bufferFilled=" << bufferFilled; fillBuffer(&m_buffer, m_bufferRect.topLeft(), p->rect()); } #ifdef SHOW_ME_WHAT_YOU_GET_IN_D_CANVA t2 = t.elapsed(); t.start(); #endif int xV = p->rect().x() - m_bufferRect.x(); int yV = p->rect().y() - m_bufferRect.y(); int wV = p->rect().width(); int hV = p->rect().height(); if (hV > 0 && wV > 0) { // qp.drawImage(p->rect().x(), p->rect().y(), m_buffer, xV, yV, wV, hV); qp.drawPixmap(p->rect().x(), p->rect().y(), m_buffer, xV, yV, wV, hV); #if DRAW_DEBUG_LINES // qDebug() << "normal rendering" << xV << yV << wV << hV << "at" << p->rect().x() << p->rect().y(); qp.setPen(Qt::blue); qp.drawLine(p->rect().x(), p->rect().y(), p->rect().x() + p->rect().width(), p->rect().y() + p->rect().height()); qp.drawLine(p->rect().x() + p->rect().width(), p->rect().y(), p->rect().x(), p->rect().y() + p->rect().height()); #endif } } #ifdef SHOW_ME_WHAT_YOU_GET_IN_D_CANVA t3 = t.elapsed(); t.start(); #endif if (m_doc->appMode != modeNormal) { qp.save(); qp.scale(m_viewMode.scale, m_viewMode.scale); qp.translate(-m_doc->minCanvasCoordinate.x() * m_viewMode.scale, -m_doc->minCanvasCoordinate.y() * m_viewMode.scale); drawControls( &qp ); qp.restore(); } #ifdef SHOW_ME_WHAT_YOU_GET_IN_D_CANVA t4 = t.elapsed(); t.start(); #endif break; case RENDER_BUFFERED: { #ifdef SHOW_ME_WHAT_YOU_GET_IN_D_CANVA dmode = "BUFFER"; t1 = t.elapsed(); t.start(); #endif int xV = p->rect().x() - m_bufferRect.x(); int yV = p->rect().y() - m_bufferRect.y(); int wV = p->rect().width(); int hV = p->rect().height(); if (xV < 0) { wV += xV; xV = 0; } if (yV < 0) { hV += yV; yV = 0; } #ifdef SHOW_ME_WHAT_YOU_GET_IN_D_CANVA t2 = t.elapsed(); t.start(); #endif if (hV > 0 && wV > 0) { // qp.drawImage(p->rect().x(), p->rect().y(), m_buffer, xV, yV, wV, hV); qp.drawPixmap(p->rect().x(), p->rect().y(), m_buffer, xV, yV, wV, hV); #if DRAW_DEBUG_LINES // qDebug() << "buffered rendering" << xV << yV << wV << hV << "at" << p->rect().x() << p->rect().y(); qp.setPen(Qt::green); qp.drawLine(p->rect().x(), p->rect().y(), p->rect().x() + p->rect().width(), p->rect().y() + p->rect().height()); qp.drawLine(p->rect().x() + p->rect().width(), p->rect().y(), p->rect().x(), p->rect().y() + p->rect().height()); #endif } #ifdef SHOW_ME_WHAT_YOU_GET_IN_D_CANVA t3 = t.elapsed(); t.start(); #endif if (m_doc->appMode != modeNormal) { qp.save(); qp.scale(m_viewMode.scale, m_viewMode.scale); qp.translate(-m_doc->minCanvasCoordinate.x(), -m_doc->minCanvasCoordinate.y()); drawControls( &qp ); qp.restore(); } #ifdef SHOW_ME_WHAT_YOU_GET_IN_D_CANVA t4 = t.elapsed(); t.start(); #endif } break; case RENDER_SELECTION_SEPARATE: break; case RENDER_SELECTION_BUFFERED: break; case RENDER_LEGACY: default: assert (false); } #ifdef SHOW_ME_WHAT_YOU_GET_IN_D_CANVA t5 = t.elapsed(); t.start(); #endif // does mode specific rendering, currently selection in legacymode and nodes in nodeedit m_view->m_canvasMode->drawControls(&qp); QPainter tp(this); tp.drawPixmap(p->rect(), tmpImg, tmpImg.rect()); #ifdef SHOW_ME_WHAT_YOU_GET_IN_D_CANVA t6 = t.elapsed(); qDebug()<Pages->count(); ScPainter *painter=0; QImage img = QImage(clipw, cliph, QImage::Format_ARGB32); painter = new ScPainter(&img, img.width(), img.height(), 1.0, 0); painter->clear(palette().color(QPalette::Window)); painter->newPath(); painter->moveTo(0, 0); painter->lineTo(clipw, 0); painter->lineTo(clipw, cliph); painter->lineTo(0, cliph); painter->closePath(); painter->setClipPath(); painter->translate(-clipx, -clipy); painter->setZoomFactor(m_viewMode.scale); painter->translate(-m_doc->minCanvasCoordinate.x(), -m_doc->minCanvasCoordinate.y()); painter->setLineWidth(1); painter->setFillMode(ScPainter::Solid); // Tsetup = tim.elapsed(); if (!m_doc->masterPageMode()) { drawBackgroundPageOutlines(painter, clipx, clipy, clipw, cliph); // Toutlines= tim.elapsed(); if (m_viewMode.viewAsPreview) { FPointArray PoLine; getClipPathForPages(&PoLine); painter->beginLayer(1.0, 0, &PoLine); } else painter->beginLayer(1.0, 0); if ((m_doc->guidesSettings.before) && (!m_viewMode.viewAsPreview)) { drawGuides(painter, clipx, clipy, clipw, cliph); } for (uint a = 0; a < docPagesCount; ++a) { DrawMasterItems(painter, m_doc->Pages->at(a), QRect(clipx, clipy, clipw, cliph)); } DrawPageItems(painter, QRect(clipx, clipy, clipw, cliph)); painter->endLayer(); // Tcontents = tim.elapsed(); if ((!m_doc->guidesSettings.before) && (!m_viewMode.viewAsPreview)) { drawGuides(painter, clipx, clipy, clipw, cliph); } } else // masterPageMode { drawBackgroundMasterpage(painter, clipx, clipy, clipw, cliph); // Tbackground = tim.elapsed(); painter->beginLayer(1.0, 0); DrawPageItems(painter, QRect(clipx, clipy, clipw, cliph)); painter->endLayer(); // Tcontents = tim.elapsed(); double x = m_doc->scratch.Left * m_viewMode.scale; double y = m_doc->scratch.Top * m_viewMode.scale; double w = m_doc->currentPage()->width() * m_viewMode.scale; double h = m_doc->currentPage()->height() * m_viewMode.scale; QRectF drawRect = QRectF(x, y, w+5, h+5); drawRect.translate(-m_doc->minCanvasCoordinate.x() * m_viewMode.scale, -m_doc->minCanvasCoordinate.y() * m_viewMode.scale); if ((!m_doc->guidesSettings.before) && (drawRect.intersects(QRect(clipx, clipy, clipw, cliph)))) DrawPageMarks(painter, m_doc->currentPage(), QRect(clipx, clipy, clipw, cliph)); } if (((m_doc->m_Selection->count() != 0) || (m_viewMode.linkedFramesToShow.count() != 0)) && (!m_viewMode.viewAsPreview)) { drawFrameLinks(painter); } painter->end(); psx->drawImage(clipx, clipy, img); delete painter; painter=NULL; // qDebug( "Time elapsed: %d ms, setup=%d, outlines=%d, background=%d, contents=%d, rest=%d", tim.elapsed(), Tsetup,Toutlines -Tsetup, Tbackground-Toutlines, Tcontents-Tbackground, tim.elapsed() - Tcontents ); } void Canvas::drawControls(QPainter *psx) { /*if (redrawMode == 1) // move ruler origin... { pp.resetMatrix(); pp.setBrush(Qt::NoBrush); pp.setPen(QPen(Qt::black, 1.0, Qt::DotLine, Qt::FlatCap, Qt::MiterJoin)); QPoint nXY = redrawPolygon.point(0); pp.drawLine(0, nXY.y(), this->width(), nXY.y()); pp.drawLine(nXY.x(), 0, nXY.x(), this->height()); redrawPolygon.clear(); }*/ psx->save(); if ((m_doc->appMode == modeDrawBezierLine) && (!m_viewMode.redrawPolygon.isEmpty()) && (m_doc->m_Selection->count() != 0)) { drawControlsBezierCurve(psx, m_doc->m_Selection->itemAt(0)); } if (m_viewMode.m_MouseButtonPressed && (m_doc->appMode == modeMeasurementTool)) { drawControlsMeasurementLine(psx); } if (m_viewMode.m_MouseButtonPressed && (m_doc->appMode == modeDrawLine)) { drawControlsDrawLine(psx); } if (m_doc->appMode == modeEditGradientVectors) { drawControlsGradientVectors(psx, m_doc->m_Selection->itemAt(0)); } if (m_viewMode.operItemMoving || m_viewMode.operItemResizing) { if (m_viewMode.operItemResizing) { if (!m_viewMode.redrawPolygon.isEmpty()) { if (m_viewMode.m_MouseButtonPressed && (m_doc->appMode == modeDrawFreehandLine)) { drawControlsFreehandLine(psx); } else if (m_doc->appMode != modeDrawFreehandLine) { // qDebug() << "XXX drawControls - operItemResizing"; assert(false); // drawControlsHighlightRect(psx); } } } else { drawControlsMovingItemsRect(psx); } } else { if ((m_doc->m_Selection->count() != 0) && (m_doc->appMode != modeDrawBezierLine)) { // drawControlsSelection(psx, m_doc->m_Selection->itemAt(0)); /* PageItem *currItem = m_doc->m_Selection->itemAt(0); if ((m_doc->appMode == modeEditClip) && (currItem->isSelected())) { drawControlsNodeEditPoints(psx, currItem); } else { drawControlsSelectionSpecial(psx, currItem); }*/ } } psx->restore(); } /** */ void Canvas::drawControlsMovingItemsRect(QPainter* pp) { if (m_doc->m_Selection->count() != 0) { uint selectedItemCount = m_doc->m_Selection->count(); PageItem *currItem = m_doc->m_Selection->itemAt(0); if (selectedItemCount < moveWithBoxesOnlyThreshold) { for (uint cu = 0; cu < selectedItemCount; cu++) { currItem = m_doc->m_Selection->itemAt(cu); // pp->resetMatrix(); // QPoint out = contentsToViewport(QPoint(0, 0)); // pp->translate(out.x(), out.y()); // pp->translate(-qRound(m_doc->minCanvasCoordinate.x()*m_viewMode.scale), -qRound(m_doc->minCanvasCoordinate.y()*m_viewMode.scale)); pp->save(); // Transform(currItem, pp); pp->translate(currItem->xPos(), currItem->yPos()); pp->rotate(currItem->rotation()); pp->setBrush(Qt::NoBrush); pp->setPen(QPen(Qt::black, 1.0 / m_viewMode.scale, Qt::DotLine, Qt::FlatCap, Qt::MiterJoin)); if (selectedItemCount < moveWithFullOutlinesThreshold) { if (!(currItem->asLine())) currItem->DrawPolyL(pp, currItem->Clip); else { if (currItem->asLine()) { int lw2 = 1; int lw = 1; Qt::PenCapStyle le = Qt::FlatCap; if (currItem->NamedLStyle.isEmpty()) { if (currItem->lineColor() != CommonStrings::None) { lw2 = qRound(currItem->lineWidth() / 2.0); lw = qRound(qMax(currItem->lineWidth(), 1.0)); } le = currItem->PLineEnd; } else { multiLine ml = m_doc->MLineStyles[currItem->NamedLStyle]; lw2 = qRound(ml[ml.size()-1].Width / 2.0); lw = qRound(qMax(ml[ml.size()-1].Width, 1.0)); le = static_cast(ml[ml.size()-1].LineEnd); } if (le != Qt::FlatCap) pp->drawRect(-lw2, -lw2, qRound(currItem->width())+lw, lw); else pp->drawRect(-1, -lw2, qRound(currItem->width()), lw); } } } else pp->drawRect(0, 0, static_cast(currItem->width())+1, static_cast(currItem->height())+1); pp->restore(); } } else { double gx, gy, gw, gh; m_doc->m_Selection->setGroupRect(); m_doc->m_Selection->getGroupRect(&gx, &gy, &gw, &gh); QPoint out = contentsToViewport(QPoint(0, 0)); // pp->resetMatrix(); // pp->translate(out.x(), out.y()); // pp->translate(-qRound(m_doc->minCanvasCoordinate.x()*m_viewMode.scale), -qRound(m_doc->minCanvasCoordinate.y()*m_viewMode.scale)); // pp->scale(m_viewMode.scale, m_viewMode.scale); pp->translate(gx, gy); pp->drawRect(QRectF(0.0, 0.0, gw, gh)); } } } /** void Canvas::drawControlsHighlightRect(QPainter* pp) { QColor drawColor = qApp->palette().color(QPalette::Active, QPalette::Highlight); pp->setPen(QPen(drawColor, 1, Qt::SolidLine, Qt::FlatCap, Qt::MiterJoin)); drawColor.setAlpha(64); // pp->resetMatrix(); // QPoint out = contentsToViewport(QPoint(0, 0)); // pp->translate(out.x(), out.y()); // pp->translate(-qRound(m_doc->minCanvasCoordinate.x()*m_viewMode.scale), -qRound(m_doc->minCanvasCoordinate.y()*m_viewMode.scale)); pp->setBrush(drawColor); pp->drawPolygon(m_viewMode.redrawPolygon); m_viewMode.redrawPolygon.clear(); } */ /** */ void Canvas::drawControlsGradientVectors(QPainter* psx, PageItem *currItem) { // psx->resetMatrix(); // QPoint out = contentsToViewport(QPoint(0, 0)); // psx->translate(out.x(), out.y()); // psx->translate(-qRound(m_doc->minCanvasCoordinate.x()*m_viewMode.scale), -qRound(m_doc->minCanvasCoordinate.y()*m_viewMode.scale)); //Transform(currItem, psx); psx->translate(static_cast(currItem->xPos()), static_cast(currItem->yPos())); psx->rotate(currItem->rotation()); psx->setPen(QPen(Qt::blue, 1.0 / m_viewMode.scale, Qt::SolidLine, Qt::FlatCap, Qt::MiterJoin)); psx->setBrush(Qt::NoBrush); psx->drawLine(QPointF(currItem->GrStartX, currItem->GrStartY), QPointF(currItem->GrEndX, currItem->GrEndY)); psx->setPen(QPen(Qt::magenta, 8.0 / m_viewMode.scale, Qt::SolidLine, Qt::RoundCap, Qt::MiterJoin)); psx->drawPoint(QPointF(currItem->GrStartX, currItem->GrStartY)); psx->drawPoint(QPointF(currItem->GrEndX, currItem->GrEndY)); } /** draws the bezier curve in edit bezier mode */ void Canvas::drawControlsBezierCurve(QPainter* pp, PageItem* currItem) { pp->setBrush(Qt::NoBrush); pp->setPen(QPen(Qt::black, 1.0 / m_viewMode.scale, Qt::DotLine, Qt::FlatCap, Qt::MiterJoin)); pp->translate(static_cast(currItem->xPos()), static_cast(currItem->yPos())); pp->rotate(currItem->rotation()); QPainterPath Bez; if (currItem->PoLine.size() > 1) { QPoint nXY = m_viewMode.redrawPolygon.point(0); if (!m_viewMode.m_MouseButtonPressed) { FPoint a1 = currItem->PoLine.point(currItem->PoLine.size()-2); FPoint a2 = currItem->PoLine.point(currItem->PoLine.size()-1); Bez.moveTo(a1.x(), a1.y()); Bez.cubicTo(a2.x(), a2.y(), nXY.x(), nXY.y(), nXY.x(), nXY.y()); pp->drawPath(Bez); } else { FPoint a2 = currItem->PoLine.point(currItem->PoLine.size()-1); if (currItem->PoLine.size() > 2) { FPoint a1 = currItem->PoLine.point(currItem->PoLine.size()-2); FPoint a3 = currItem->PoLine.point(currItem->PoLine.size()-3); Bez.moveTo(a3.x(), a3.y()); Bez.cubicTo(a1.x(), a1.y(), nXY.x(), nXY.y(), a2.x(), a2.y()); pp->drawPath(Bez); } pp->drawLine(QPoint(qRound(a2.x()), qRound(a2.y())), nXY); } } else { QPoint a2 = currItem->PoLine.pointQ(currItem->PoLine.size()-1); QPoint nXY = m_viewMode.redrawPolygon.point(0); pp->drawLine(a2, nXY); } m_viewMode.redrawPolygon.clear(); } /** draws the measurement line in measure mode */ void Canvas::drawControlsMeasurementLine(QPainter* pp) { // pp->resetMatrix(); // QPoint out = contentsToViewport(QPoint(0, 0)); // pp->translate(out.x(), out.y()); pp->setBrush(Qt::NoBrush); pp->setPen(QPen(Qt::black, 1.0, Qt::DotLine, Qt::FlatCap, Qt::MiterJoin)); pp->drawPolyline(m_viewMode.redrawPolygon); m_viewMode.redrawPolygon.clear(); } /** draws the line in drawline mode */ void Canvas::drawControlsDrawLine(QPainter* pp) { // QPoint out = contentsToViewport(QPoint(0, 0)); // pp->resetMatrix(); // pp->translate(out.x(), out.y()); // pp->translate(-qRound(m_doc->minCanvasCoordinate.x()*m_viewMode.scale), -qRound(m_doc->minCanvasCoordinate.y()*m_viewMode.scale)); // pp->scale(m_viewMode.scale, m_viewMode.scale); pp->setBrush(Qt::NoBrush); pp->setPen(QPen(Qt::black, 1.0 / m_viewMode.scale, Qt::DotLine, Qt::FlatCap, Qt::MiterJoin)); pp->drawPolyline(m_viewMode.redrawPolygon); m_viewMode.redrawPolygon.clear(); } /** draws the freehand curve in freehand mode */ void Canvas::drawControlsFreehandLine(QPainter* pp) { // pp->resetMatrix(); // QPoint out = contentsToViewport(QPoint(0, 0)); // pp->translate(out.x(), out.y()); // pp->translate(-qRound(m_doc->minCanvasCoordinate.x()*m_viewMode.scale), -qRound(m_doc->minCanvasCoordinate.y()*m_viewMode.scale)); // pp->scale(m_viewMode.scale, m_viewMode.scale); pp->setBrush(Qt::NoBrush); pp->setPen(QPen(Qt::black, 1.0 / m_viewMode.scale, Qt::DotLine, Qt::FlatCap, Qt::MiterJoin)); pp->drawPolyline(m_viewMode.redrawPolygon); m_viewMode.redrawPolygon.clear(); } /** draws masterpage items in layer order */ void Canvas::DrawMasterItems(ScPainter *painter, Page *page, QRect clip) { FPoint orig = localToCanvas(clip.topLeft()); QRectF cullingArea = QRectF(static_cast(orig.x()), static_cast(orig.y()), qRound(clip.width() / m_viewMode.scale + 0.5), qRound(clip.height() / m_viewMode.scale + 0.5)); QStack groupStack; QStack groupStack2; if (!page->MPageNam.isEmpty()) { Page* Mp = m_doc->MasterPages.at(m_doc->MasterNames[page->MPageNam]); if (page->FromMaster.count() != 0) { int Lnr; ScLayer ll; PageItem *currItem; ll.isViewable = false; ll.LNr = 0; Lnr = 0; uint layerCount=m_doc->layerCount(); for (uint la = 0; la < layerCount; ++la) { m_doc->Layers.levelToLayer(ll, Lnr); bool pr = true; if ((m_viewMode.previewMode) && (!ll.isPrintable)) pr = false; if ((m_viewMode.viewAsPreview) && (!ll.isPrintable)) pr = false; if ((ll.isViewable) && (pr)) { if ((layerCount > 1) && ((ll.blendMode != 0) || (ll.transparency != 1.0)) && (!ll.outlineMode)) painter->beginLayer(ll.transparency, ll.blendMode); uint pageFromMasterCount=page->FromMaster.count(); for (uint a = 0; a < pageFromMasterCount; ++a) { currItem = page->FromMaster.at(a); if (currItem->LayerNr != ll.LNr) continue; if ((currItem->OwnPage != -1) && (currItem->OwnPage != static_cast(Mp->pageNr()))) continue; if ((m_viewMode.previewMode) && (!currItem->printEnabled())) continue; if ((m_viewMode.viewAsPreview) && (!currItem->printEnabled())) continue; double OldX = currItem->xPos(); double OldY = currItem->yPos(); double OldBX = currItem->BoundingX; double OldBY = currItem->BoundingY; if (!currItem->ChangedMasterItem) { //Hack to not check for undo changes, indicate drawing only currItem->moveBy(-Mp->xOffset() + page->xOffset(), -Mp->yOffset() + page->yOffset(), true); currItem->BoundingX = OldBX - Mp->xOffset() + page->xOffset(); currItem->BoundingY = OldBY - Mp->yOffset() + page->yOffset(); } if (currItem->isGroupControl) { painter->save(); currItem->savedOwnPage = currItem->OwnPage; currItem->OwnPage = page->pageNr(); if ((cullingArea.intersects(currItem->getBoundingRect().adjusted(0.0, 0.0, 1.0, 1.0))) && (m_doc->guidesSettings.layerMarkersShown) && (m_doc->layerCount() > 1)) currItem->DrawObj(painter, cullingArea); FPointArray cl = currItem->PoLine.copy(); QMatrix mm; mm.translate(currItem->xPos(), currItem->yPos()); mm.rotate(currItem->rotation()); cl.map( mm ); painter->beginLayer(1.0 - currItem->fillTransparency(), currItem->fillBlendmode(), &cl); groupStack.push(currItem->groupsLastItem); groupStack2.push(currItem); currItem->OwnPage = currItem->savedOwnPage; if (!currItem->ChangedMasterItem) { //Hack to not check for undo changes, indicate drawing only currItem->setXYPos(OldX, OldY, true); currItem->BoundingX = OldBX; currItem->BoundingY = OldBY; } continue; } currItem->savedOwnPage = currItem->OwnPage; currItem->OwnPage = page->pageNr(); //FIXME if (!evSpon || forceRedraw) // currItem->invalid = true; if (cullingArea.intersects(currItem->getBoundingRect().adjusted(0.0, 0.0, 1.0, 1.0))) { if (!((m_viewMode.operItemMoving) && (currItem->isSelected()))) { if (m_viewMode.forceRedraw) currItem->invalidateLayout(); currItem->DrawObj(painter, cullingArea); } // else // qDebug() << "skip masterpage item (move/resizeEdit/selected)" << m_viewMode.operItemMoving << currItem->isSelected(); } currItem->OwnPage = currItem->savedOwnPage; if (!currItem->ChangedMasterItem) { //Hack to not check for undo changes, indicate drawing only currItem->setXYPos(OldX, OldY, true); currItem->BoundingX = OldBX; currItem->BoundingY = OldBY; } if (groupStack.count() != 0) { while (currItem == groupStack.top()) { painter->endLayer(); painter->restore(); PageItem *cite = groupStack2.pop(); double OldX = cite->xPos(); double OldY = cite->yPos(); double OldBX = cite->BoundingX; double OldBY = cite->BoundingY; if (!cite->ChangedMasterItem) { //Hack to not check for undo changes, indicate drawing only cite->moveBy(-Mp->xOffset() + page->xOffset(), -Mp->yOffset() + page->yOffset(), true); cite->BoundingX = OldBX - Mp->xOffset() + page->xOffset(); cite->BoundingY = OldBY - Mp->yOffset() + page->yOffset(); } if ((cullingArea.intersects(cite->getBoundingRect().adjusted(0.0, 0.0, 1.0, 1.0))) && (m_doc->guidesSettings.layerMarkersShown) && (m_doc->layerCount() > 1)) cite->DrawObj(painter, cullingArea); cite->OwnPage = cite->savedOwnPage; if (!currItem->ChangedMasterItem) { //Hack to not check for undo changes, indicate drawing only cite->setXYPos(OldX, OldY, true); cite->BoundingX = OldBX; cite->BoundingY = OldBY; } groupStack.pop(); if (groupStack.count() == 0) break; } } } for (uint a = 0; a < pageFromMasterCount; ++a) { currItem = page->FromMaster.at(a); if (currItem->LayerNr != ll.LNr) continue; if (!currItem->isTableItem) continue; if ((m_viewMode.previewMode) && (!currItem->printEnabled())) continue; if ((m_viewMode.viewAsPreview) && (!currItem->printEnabled())) continue; if ((currItem->OwnPage != -1) && (currItem->OwnPage != static_cast(Mp->pageNr()))) continue; if (currItem->isGroupControl) continue; double OldX = currItem->xPos(); double OldY = currItem->yPos(); double OldBX = currItem->BoundingX; double OldBY = currItem->BoundingY; if (!currItem->ChangedMasterItem) { //Hack to not check for undo changes, indicate drawing only currItem->setXYPos(OldX - Mp->xOffset() + page->xOffset(), OldY - Mp->yOffset() + page->yOffset(), true); currItem->BoundingX = OldBX - Mp->xOffset() + page->xOffset(); currItem->BoundingY = OldBY - Mp->yOffset() + page->yOffset(); } if (cullingArea.intersects(currItem->getBoundingRect().adjusted(0.0, 0.0, 1.0, 1.0))) { painter->save(); painter->translate(currItem->xPos(), currItem->yPos()); painter->rotate(currItem->rotation()); if ((currItem->lineColor() != CommonStrings::None) && (currItem->lineWidth() != 0.0)) { QColor tmp; currItem->SetQColor(&tmp, currItem->lineColor(), currItem->lineShade()); if ((currItem->TopLine) || (currItem->RightLine) || (currItem->BottomLine) || (currItem->LeftLine)) { painter->setPen(tmp, currItem->lineWidth(), currItem->PLineArt, Qt::SquareCap, currItem->PLineJoin); if (currItem->TopLine) painter->drawLine(FPoint(0.0, 0.0), FPoint(currItem->width(), 0.0)); if (currItem->RightLine) painter->drawLine(FPoint(currItem->width(), 0.0), FPoint(currItem->width(), currItem->height())); if (currItem->BottomLine) painter->drawLine(FPoint(currItem->width(), currItem->height()), FPoint(0.0, currItem->height())); if (currItem->LeftLine) painter->drawLine(FPoint(0.0, currItem->height()), FPoint(0.0, 0.0)); } } painter->restore(); } if (!currItem->ChangedMasterItem) { //Hack to not check for undo changes, indicate drawing only currItem->setXYPos(OldX, OldY, true); currItem->BoundingX = OldBX; currItem->BoundingY = OldBY; } } if ((layerCount > 1) && ((ll.blendMode != 0) || (ll.transparency != 1.0)) && (!ll.outlineMode)) painter->endLayer(); } Lnr++; } } } } /** draws page items in layer order */ void Canvas::DrawPageItems(ScPainter *painter, QRect clip) { // qDebug()<<"Canvas::DrawPageItems"<(orig.x()), static_cast(orig.y()), qRound(clip.width() / m_viewMode.scale + 0.5), qRound(clip.height() / m_viewMode.scale + 0.5)); QStack groupStack; QStack groupStack2; if (m_doc->Items->count() != 0) { int Lnr=0; ScLayer ll; PageItem *currItem; ll.isViewable = false; ll.LNr = 0; uint layerCount=m_doc->layerCount(); int docCurrPageNo=static_cast(m_doc->currentPageNumber()); for (uint la2 = 0; la2 < layerCount; ++la2) { m_doc->Layers.levelToLayer(ll, Lnr); bool pr = true; if ((m_viewMode.previewMode) && (!ll.isPrintable)) pr = false; if ((m_viewMode.viewAsPreview) && (!ll.isPrintable)) pr = false; if ((ll.isViewable) && (pr)) { if ((layerCount > 1) && ((ll.blendMode != 0) || (ll.transparency != 1.0)) && (!ll.outlineMode)) painter->beginLayer(ll.transparency, ll.blendMode); for (int it = 0; it < m_doc->Items->count(); ++it) { currItem = m_doc->Items->at(it); if (currItem->LayerNr != ll.LNr) continue; if ((m_viewMode.previewMode) && (!currItem->printEnabled())) continue; if ((m_viewMode.viewAsPreview) && (!currItem->printEnabled())) continue; if ((m_doc->masterPageMode()) && ((currItem->OwnPage != -1) && (currItem->OwnPage != docCurrPageNo))) continue; if (!m_doc->masterPageMode() && !currItem->OnMasterPage.isEmpty()) { if (currItem->OnMasterPage != m_doc->currentPage()->pageName()) continue; } if (currItem->isGroupControl) { painter->save(); FPointArray cl = currItem->PoLine.copy(); QMatrix mm; mm.translate(currItem->xPos(), currItem->yPos()); mm.rotate(currItem->rotation()); cl.map( mm ); painter->beginLayer(1.0 - currItem->fillTransparency(), currItem->fillBlendmode(), &cl); groupStack.push(currItem->groupsLastItem); groupStack2.push(currItem); continue; } if (cullingArea.intersects(currItem->getBoundingRect().adjusted(0.0, 0.0, 1.0, 1.0))) { //FIXME if (!evSpon || forceRedraw) // currItem->invalid = true; // if ((!m_MouseButtonPressed) || (m_doc->appMode == modeEditClip)) if (((m_viewMode.operItemMoving || m_viewMode.drawSelectedItemsWithControls) && currItem->isSelected())) { // qDebug() << "skipping pageitem (move/resizeEdit/selected)" << m_viewMode.operItemMoving << currItem->isSelected(); } else if(m_viewMode.operItemSelecting) { currItem->invalid = false; currItem->DrawObj(painter, cullingArea); } else { // I comment it because the "view" should not // alter the "data". And it really prevents optimisation - pm // if (m_viewMode.forceRedraw) // currItem->invalidateLayout(); currItem->DrawObj(painter, cullingArea); } // currItem->Redrawn = true; if ((currItem->asTextFrame()) && ((currItem->nextInChain() != 0) || (currItem->prevInChain() != 0))) { PageItem *nextItem = currItem; while (nextItem != 0) { if (nextItem->prevInChain() != 0) nextItem = nextItem->prevInChain(); else break; } if (!m_viewMode.linkedFramesToShow.contains(nextItem)) m_viewMode.linkedFramesToShow.append(nextItem); } /* FIXME:av - what to fix exactly? - pm */ if ((m_doc->appMode == modeEdit) && (currItem->isSelected()) && (currItem->itemType() == PageItem::TextFrame)) { setupEditHRuler(currItem); } } if (groupStack.count() != 0) { while (currItem == groupStack.top()) { painter->endLayer(); painter->restore(); PageItem *cite = groupStack2.pop(); if ((cullingArea.intersects(cite->getBoundingRect())) && (((m_doc->guidesSettings.layerMarkersShown) && (m_doc->layerCount() > 1)) || (cite->textFlowUsesContourLine()))) cite->DrawObj(painter, cullingArea); groupStack.pop(); if (groupStack.count() == 0) break; } } } for (int it = 0; it < m_doc->Items->count(); ++it) { currItem = m_doc->Items->at(it); if (currItem->LayerNr != ll.LNr) continue; if (!currItem->isTableItem) continue; if ((m_viewMode.previewMode) && (!currItem->printEnabled())) continue; if ((m_viewMode.viewAsPreview) && (!currItem->printEnabled())) continue; if (cullingArea.intersects(currItem->getBoundingRect().adjusted(0.0, 0.0, 1.0, 1.0))) { painter->save(); painter->translate(currItem->xPos(), currItem->yPos()); painter->rotate(currItem->rotation()); if ((currItem->lineColor() != CommonStrings::None) && (currItem->lineWidth() != 0.0)) { QColor tmp; currItem->SetQColor(&tmp, currItem->lineColor(), currItem->lineShade()); if ((currItem->TopLine) || (currItem->RightLine) || (currItem->BottomLine) || (currItem->LeftLine)) { painter->setPen(tmp, currItem->lineWidth(), currItem->PLineArt, Qt::SquareCap, currItem->PLineJoin); if (currItem->TopLine) painter->drawLine(FPoint(0.0, 0.0), FPoint(currItem->width(), 0.0)); if (currItem->RightLine) painter->drawLine(FPoint(currItem->width(), 0.0), FPoint(currItem->width(), currItem->height())); if (currItem->BottomLine) painter->drawLine(FPoint(currItem->width(), currItem->height()), FPoint(0.0, currItem->height())); if (currItem->LeftLine) painter->drawLine(FPoint(0.0, currItem->height()), FPoint(0.0, 0.0)); } } painter->restore(); } } if ((layerCount > 1) && ((ll.blendMode != 0) || (ll.transparency != 1.0)) && (!ll.outlineMode)) painter->endLayer(); } Lnr++; } } } /** Draws the canvas background for masterpages, incl. bleeds */ void Canvas::drawBackgroundMasterpage(ScPainter* painter, int clipx, int clipy, int clipw, int cliph) { double x = m_doc->scratch.Left * m_viewMode.scale; double y = m_doc->scratch.Top * m_viewMode.scale; double w = m_doc->currentPage()->width() * m_viewMode.scale; double h = m_doc->currentPage()->height() * m_viewMode.scale; QRectF drawRect = QRectF(x, y, w+5, h+5); drawRect.translate(-m_doc->minCanvasCoordinate.x(), -m_doc->minCanvasCoordinate.y()); if (drawRect.intersects(QRectF(clipx, clipy, clipw, cliph))) { painter->setFillMode(ScPainter::Solid); painter->setBrush(QColor(128,128,128)); MarginStruct pageBleeds; m_doc->getBleeds(m_doc->currentPage(), pageBleeds); // painter->beginLayer(1.0, 0); painter->setAntialiasing(false); painter->setPen(Qt::black, 1 / m_viewMode.scale, Qt::SolidLine, Qt::FlatCap, Qt::MiterJoin); if (m_doc->bleeds.hasNonZeroValue() && m_doc->guidesSettings.showBleed) { // painter->drawRect(m_doc->scratch.Left - bleedLeft+5 / m_viewMode.scale, m_doc->scratch.Top - bleedTop+5 / m_viewMode.scale, m_doc->currentPage()->width() + bleedLeft + bleedRight, m_doc->currentPage()->height() + bleedBottom + bleedTop); if (PrefsManager::instance()->appPrefs.showPageShadow) painter->drawRect(m_doc->scratch.Left - pageBleeds.Left+5, m_doc->scratch.Top - pageBleeds.Top+5, m_doc->currentPage()->width() + pageBleeds.Left + pageBleeds.Right, m_doc->currentPage()->height() + pageBleeds.Bottom + pageBleeds.Top); painter->setBrush(m_doc->papColor); painter->drawRect(m_doc->scratch.Left - pageBleeds.Left, m_doc->scratch.Top - pageBleeds.Top, m_doc->currentPage()->width() + pageBleeds.Left + pageBleeds.Right, m_doc->currentPage()->height() + pageBleeds.Bottom + pageBleeds.Top); } else { // painter->drawRect(m_doc->scratch.Left+5 / m_viewMode.scale, m_doc->scratch.Top+5 / m_viewMode.scale, m_doc->currentPage()->width(), m_doc->currentPage()->height()); if (PrefsManager::instance()->appPrefs.showPageShadow) painter->drawRect(m_doc->scratch.Left+5, m_doc->scratch.Top+5, m_doc->currentPage()->width(), m_doc->currentPage()->height()); painter->setBrush(m_doc->papColor); painter->drawRect(m_doc->scratch.Left, m_doc->scratch.Top, m_doc->currentPage()->width(), m_doc->currentPage()->height()); } painter->setAntialiasing(true); if (m_doc->guidesSettings.before) DrawPageMarks(painter, m_doc->currentPage(), QRect(clipx, clipy, clipw, cliph)); // painter->endLayer(); } } /** draws the page outlines on canvas, including bleed area */ void Canvas::drawBackgroundPageOutlines(ScPainter* painter, int clipx, int clipy, int clipw, int cliph) { // qDebug()<<"Canvas::drawBackgroundPageOutlines"<< clipx<< clipy<< clipw<Pages->count(); if (PrefsManager::instance()->appPrefs.showPageShadow && !m_viewMode.viewAsPreview) { painter->setBrush(QColor(128,128,128)); painter->setAntialiasing(false); // painter->beginLayer(1.0, 0); painter->setPen(Qt::black, 1.0 / m_viewMode.scale, Qt::SolidLine, Qt::FlatCap, Qt::MiterJoin); Page *actPg; MarginStruct pageBleeds; for (int a = 0; a < static_cast(docPagesCount); ++a) { actPg = m_doc->Pages->at(a); m_doc->getBleeds(actPg, pageBleeds); double blx = (actPg->xOffset() - pageBleeds.Left) * m_viewMode.scale; double bly = (actPg->yOffset() - pageBleeds.Top) * m_viewMode.scale; double blw = (actPg->width() + pageBleeds.Left + pageBleeds.Right) * m_viewMode.scale; double blh = (actPg->height() + pageBleeds.Bottom + pageBleeds.Top) * m_viewMode.scale; QRectF drawRect = QRectF(blx-1, bly-1, blw+6, blh+6); drawRect.translate(-m_doc->minCanvasCoordinate.x() * m_viewMode.scale, -m_doc->minCanvasCoordinate.y() * m_viewMode.scale); if (drawRect.intersects(QRectF(clipx, clipy, clipw, cliph))) { painter->setFillMode(ScPainter::Solid); double blx2 = actPg->xOffset(); double bly2 = actPg->yOffset(); double blw2 = actPg->width(); double blh2 = actPg->height(); if (m_doc->guidesSettings.showBleed) { blx2 -= pageBleeds.Left; bly2 -= pageBleeds.Top; blw2 += pageBleeds.Left + pageBleeds.Right; blh2 += pageBleeds.Bottom + pageBleeds.Top; } // painter->drawRect(blx2 + 5 /* m_viewMode.scale */, bly2 + 5 /* m_viewMode.scale */, blw2, blh2); painter->drawRect(blx2 + 5, bly2 + 5, blw2, blh2); if (m_doc->bleeds.hasNonZeroValue() && m_doc->guidesSettings.showBleed) { painter->setFillMode(ScPainter::None); painter->setPen(Qt::black, 1.0 / m_viewMode.scale, Qt::SolidLine, Qt::FlatCap, Qt::MiterJoin); painter->drawRect(blx2 - 1 / m_viewMode.scale, bly2 - 1 / m_viewMode.scale, blw2 + 2 / m_viewMode.scale, blh2 + 2 / m_viewMode.scale); } } } // painter->endLayer(); painter->setAntialiasing(true); } painter->setFillMode(ScPainter::Solid); Page *actPg; MarginStruct pageBleeds; for (int a = 0; a < static_cast(docPagesCount); ++a) { actPg = m_doc->Pages->at(a); double x = actPg->xOffset(); double y = actPg->yOffset(); double w = actPg->width(); double h = actPg->height(); bool drawBleed = false; if (((m_doc->bleeds.Bottom != 0.0) || (m_doc->bleeds.Top != 0.0) || (m_doc->bleeds.Left != 0.0) || (m_doc->bleeds.Right != 0.0)) && (m_doc->guidesSettings.showBleed)) { drawBleed = true; m_doc->getBleeds(a, pageBleeds); } else pageBleeds.resetToZero(); double blx = (actPg->xOffset() - pageBleeds.Left) * m_viewMode.scale; double bly = (actPg->yOffset() - pageBleeds.Top) * m_viewMode.scale; double blw = (actPg->width() + pageBleeds.Left + pageBleeds.Right) * m_viewMode.scale; double blh = (actPg->height() + pageBleeds.Bottom + pageBleeds.Top) * m_viewMode.scale; QRectF drawRect = QRectF(blx, bly, blw+5, blh+5); drawRect.translate(-m_doc->minCanvasCoordinate.x() * m_viewMode.scale, -m_doc->minCanvasCoordinate.y() * m_viewMode.scale); if (drawRect.intersects(QRectF(clipx, clipy, clipw, cliph))) { painter->setFillMode(ScPainter::Solid); painter->setPen(Qt::black, 0, Qt::SolidLine, Qt::FlatCap, Qt::MiterJoin); painter->setAntialiasing(false); // painter->beginLayer(1.0, 0); painter->setLineWidth(0.0); painter->setBrush(m_doc->papColor); if (!m_viewMode.viewAsPreview) { double blx2 = actPg->xOffset() - pageBleeds.Left; double bly2 = actPg->yOffset() - pageBleeds.Top; double blw2 = actPg->width() + pageBleeds.Left + pageBleeds.Right; double blh2 = actPg->height() + pageBleeds.Bottom + pageBleeds.Top; painter->drawRect(blx2, bly2, blw2, blh2); if (drawBleed) painter->drawRect(x, y, w, h); } else painter->drawRect(x, y, w, h); // painter->endLayer(); painter->setAntialiasing(true); } } } /** fills PoLine with a clippath for the doc pages */ void Canvas::getClipPathForPages(FPointArray* PoLine) { PoLine->resize(0); uint docPagesCount=m_doc->Pages->count(); bool first = true; Page *actPg; for (int a = 0; a < static_cast(docPagesCount); ++a) { if (!first) PoLine->setMarker(); first = false; actPg = m_doc->Pages->at(a); double x = actPg->xOffset(); double y = actPg->yOffset(); double w = actPg->width(); double h = actPg->height(); static double rect[] = {0.0, 0.0, 0.0, 0.0, 1.0, 0.0, 1.0, 0.0, 1.0, 0.0, 1.0, 0.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 1.0, 0.0, 1.0, 0.0, 1.0, 0.0, 1.0, 0.0, 1.0, 0.0, 0.0, 0.0, 0.0}; for (int a = 0; a < 29; a += 4) { PoLine->addPoint(x + w * rect[a], y + h * rect[a+1]); PoLine->addPoint(x + w * rect[a+2], y + h * rect[a+3]); } } } void Canvas::drawGuides(ScPainter* painter, int clipx, int clipy, int clipw, int cliph) { uint docPagesCount=m_doc->Pages->count(); Page *actPg; for (uint a = 0; a < docPagesCount; ++a) { actPg = m_doc->Pages->at(a); double x = actPg->xOffset() * m_viewMode.scale; double y = actPg->yOffset() * m_viewMode.scale; double w = actPg->width() * m_viewMode.scale; double h = actPg->height() * m_viewMode.scale; QRectF drawRect = QRectF(x, y, w+5, h+5); drawRect.translate(-m_doc->minCanvasCoordinate.x() * m_viewMode.scale, -m_doc->minCanvasCoordinate.y() * m_viewMode.scale); if (drawRect.intersects(QRectF(clipx, clipy, clipw, cliph))) DrawPageMarks(painter, m_doc->Pages->at(a), QRect(clipx, clipy, clipw, cliph)); } } /** draws guides, margins, grid and baseline grid */ void Canvas::DrawPageMarks(ScPainter *p, Page *page, QRect clip) { p->save(); p->setAntialiasing(false); p->translate(page->xOffset(), page->yOffset()); double lineWidth = 1.0 / m_viewMode.scale; double pageHeight=page->height(); double pageWidth=page->width(); p->setFillMode(ScPainter::None); p->setPen(Qt::black, lineWidth, Qt::SolidLine, Qt::FlatCap, Qt::MiterJoin); p->drawRect(0, 0, pageWidth, pageHeight); //Draw the margins if (m_doc->guidesSettings.marginsShown) { p->setPen(m_doc->guidesSettings.margColor); if (m_doc->marginColored) { p->setFillMode(ScPainter::Solid); p->setBrush(m_doc->guidesSettings.margColor); p->drawRect(0, 0, pageWidth, page->Margins.Top); p->drawRect(0, page->Margins.Top, page->Margins.Left, pageHeight - page->Margins.Top); p->drawRect(page->Margins.Left, pageHeight - page->Margins.Bottom, pageWidth - page->Margins.Right - page->Margins.Left, page->Margins.Bottom); p->drawRect(pageWidth - page->Margins.Right, page->Margins.Top, page->Margins.Right, pageHeight-page->Margins.Top); p->setFillMode(ScPainter::None); } // p->setPen(m_doc->guidesSettings.margColor); p->setFillMode(ScPainter::None); p->drawRect(page->Margins.Left, page->Margins.Top, pageWidth - page->Margins.Left - page->Margins.Right, pageHeight - page->Margins.Top - page->Margins.Bottom); // p->drawLine(FPoint(0, page->Margins.Top), FPoint(pageWidth, page->Margins.Top)); // p->drawLine(FPoint(0, pageHeight - page->Margins.Bottom), FPoint(pageWidth, pageHeight - page->Margins.Bottom)); // p->drawLine(FPoint(page->Margins.Left, 0), FPoint(page->Margins.Left, pageHeight)); // p->drawLine(FPoint(pageWidth - page->Margins.Right, 0), FPoint(pageWidth - page->Margins.Right, pageHeight)); } //Draw the baseline grid if (m_doc->guidesSettings.baseShown) { p->setPen(m_doc->guidesSettings.baseColor, lineWidth, Qt::SolidLine, Qt::FlatCap, Qt::MiterJoin); for (double yg = m_doc->typographicSettings.offsetBaseGrid; yg < pageHeight; yg += m_doc->typographicSettings.valueBaseGrid) p->drawLine(FPoint(0, yg), FPoint(pageWidth, yg)); } //Draw the grid lines if (m_doc->guidesSettings.gridShown) { double lowerBx = qMax(clip.x() / m_viewMode.scale + m_doc->minCanvasCoordinate.x() - page->xOffset(), 0.0); double lowerBy = qMax(clip.y() / m_viewMode.scale + m_doc->minCanvasCoordinate.y() - page->yOffset(), 0.0); double highBx = qMin(lowerBx + clip.width() / m_viewMode.scale, pageWidth); double highBy = qMin(lowerBy + clip.height() / m_viewMode.scale, pageHeight); if (m_viewMode.scale > 0.49) { double i,start; i = m_doc->guidesSettings.majorGrid; p->setPen(m_doc->guidesSettings.majorColor, lineWidth, Qt::SolidLine, Qt::FlatCap, Qt::MiterJoin); start=floor(lowerBy/i); start*=i; for (double b = start; b <= highBy; b+=i) { p->drawLine(FPoint(qMax(lowerBx, 0.0), b), FPoint(qMin(pageWidth, highBx), b)); } start=floor(lowerBx/i); start*=i; for (double b = start; b <= highBx; b+=i) { p->drawLine(FPoint(b, qMax(lowerBy, 0.0)), FPoint(b, qMin(pageHeight, highBy))); } i = m_doc->guidesSettings.minorGrid; p->setPen(m_doc->guidesSettings.minorColor, lineWidth, Qt::DotLine, Qt::FlatCap, Qt::MiterJoin); start=floor(lowerBy/i); start*=i; for (double b = start; b <= highBy; b+=i) { p->drawLine(FPoint(qMax(lowerBx, 0.0), b), FPoint(qMin(pageWidth, highBx), b)); } start=floor(lowerBx/i); start*=i; for (double b = start; b <= highBx; b+=i) { p->drawLine(FPoint(b, qMax(lowerBy, 0.0)), FPoint(b, qMin(pageHeight, highBy))); } } } //Draw the guides if (m_doc->guidesSettings.guidesShown) page->guides.drawPage(p, m_doc, lineWidth); if (m_doc->currentPage() == page) { p->setPen(PrefsManager::instance()->appPrefs.DPageBorderColor, 2 / m_viewMode.scale, Qt::SolidLine, Qt::FlatCap, Qt::MiterJoin); p->drawRect(0, 0, pageWidth, pageHeight); } p->setAntialiasing(true); p->restore(); } /** draws the links between textframe chains. needs the list of visible textframes in m_viewMode.linkedFramesToShow */ void Canvas::drawFrameLinks(ScPainter* painter) { painter->save(); PageItem *currItem; if ((m_doc->guidesSettings.linkShown || m_viewMode.drawFramelinksWithContents) && (m_viewMode.linkedFramesToShow.count() != 0)) currItem = m_viewMode.linkedFramesToShow.at(0); else { if (m_viewMode.linkedFramesToShow.count() != 0) currItem = m_viewMode.linkedFramesToShow.at(0); else currItem = m_doc->m_Selection->itemAt(0); } //Draw the frame links if ((((m_doc->appMode == modeLinkFrames) || (m_doc->appMode == modeUnlinkFrames)) && (currItem->itemType() == PageItem::TextFrame)) || (m_doc->guidesSettings.linkShown || m_viewMode.drawFramelinksWithContents)) { PageItem *nextItem = currItem; if (m_doc->guidesSettings.linkShown || m_viewMode.drawFramelinksWithContents) { for (int lks = 0; lks < m_viewMode.linkedFramesToShow.count(); ++lks) { nextItem = m_viewMode.linkedFramesToShow.at(lks); while (nextItem != 0) { if (nextItem->nextInChain() != NULL) { FPoint start, end; calculateFrameLinkPoints(nextItem, nextItem->nextInChain(), start, end); drawLinkFrameLine(painter, start, end); } nextItem = nextItem->nextInChain(); } } } else { while (nextItem != 0) { if (nextItem->prevInChain() != 0) nextItem = nextItem->prevInChain(); else break; } while (nextItem != 0) { if (nextItem->nextInChain() != NULL) { FPoint start, end; calculateFrameLinkPoints(nextItem, nextItem->nextInChain(), start, end); drawLinkFrameLine(painter, start, end); } nextItem = nextItem->nextInChain(); } } } painter->setLineWidth(1); painter->setPenOpacity(1.0); painter->restore(); } /** draws one linkline between textframes */ void Canvas::drawLinkFrameLine(ScPainter* painter, FPoint &start, FPoint &end) { //CB FIXME Add some checking that the painter is setup? Q_ASSERT(painter!=NULL); painter->setPen(Qt::black, 1.0 / m_viewMode.scale, Qt::SolidLine, Qt::FlatCap, Qt::MiterJoin); painter->setPenOpacity(1.0); painter->drawLine(start, end); QMatrix arrowTrans; arrowTrans.translate(end.x(), end.y()); double r = atan2(end.y()-start.y(), end.x()-start.x())*(180.0/M_PI); arrowTrans.rotate(r); double sc = 0.8 / m_viewMode.scale; arrowTrans.scale(sc, sc); FPointArray arrow; arrow.addQuadPoint(-12, 0, -12, 0, -12, 0, -12, 0); arrow.addQuadPoint(-15, -5, -15, -5, -15, -5, -15, -5); arrow.addQuadPoint(0, 0, 0, 0, 0, 0, 0, 0); arrow.addQuadPoint(-15, 5, -15, 5, -15, 5, -15, 5); arrow.addQuadPoint(-12, 0, -12, 0, -12, 0, -12, 0); arrow.map(arrowTrans); painter->setBrush(painter->pen()); painter->setBrushOpacity(1.0); painter->setLineWidth(0); painter->setFillMode(ScPainter::Solid); painter->setupPolygon(&arrow); painter->fillPath(); } /** this is a wrapper around repaint() */ void Canvas::PaintSizeRect(QRect newRect) { static QRect oldRect; if (!newRect.isNull()) { if (!oldRect.isNull()) newRect.unite(oldRect); repaint(newRect.adjusted(-10, -10, 20, 20)); } oldRect = newRect; } /// ??? void Canvas::PaintSizeRect(QPolygon newRect) { static QRect oldRect; QRect newR = newRect.boundingRect(); if (!newR.isNull()) { if (!oldRect.isNull()) newR.unite(oldRect); // newR.moveBy(qRound(-m_doc->minCanvasCoordinate.x() * m_viewMode.scale), qRound(-m_doc->minCanvasCoordinate.y() * m_viewMode.scale)); m_viewMode.redrawPolygon = newRect; // m_viewMode.redrawPolygon.translate(qRound(-m_doc->minCanvasCoordinate.x() * m_viewMode.scale), qRound(-m_doc->minCanvasCoordinate.y() * m_viewMode.scale)); repaint(newR.adjusted(-20, -20, 40, 40)); } oldRect = newR; } void Canvas::getGroupRectScreen(double *x, double *y, double *w, double *h) { double gx, gy, gh, gw; m_doc->m_Selection->getGroupRect(&gx, &gy, &gw, &gh); QPoint in(qRound(gx* m_viewMode.scale), qRound(gy* m_viewMode.scale)); // in -= QPoint(qRound(m_doc->minCanvasCoordinate.x() * m_viewMode.scale), qRound(m_doc->minCanvasCoordinate.y() * m_viewMode.scale)); QPoint out = contentsToViewport(in); *x = static_cast(out.x()); *y = static_cast(out.y()); *w = gw* m_viewMode.scale; *h = gh* m_viewMode.scale; } void Canvas::Transform(PageItem *currItem, QPainter *p) { p->translate(currItem->xPos()*m_viewMode.scale, currItem->yPos()*m_viewMode.scale); p->scale(m_viewMode.scale, m_viewMode.scale); p->rotate(currItem->rotation()); } void Canvas::Transform(PageItem *currItem, QMatrix& m) { m.translate(currItem->xPos()*m_viewMode.scale, currItem->yPos()*m_viewMode.scale); m.scale(m_viewMode.scale, m_viewMode.scale); m.rotate(currItem->rotation()); } void Canvas::TransformM(PageItem *currItem, QPainter *p) { if (currItem->imageFlippedH()) { p->translate(currItem->width(), 0); p->scale(-1, 1); } if (currItem->imageFlippedV()) { p->translate(0, currItem->height()); p->scale(1, -1); } } void Canvas::calculateFrameLinkPoints(PageItem *pi1, PageItem *pi2, FPoint & start, FPoint & end) { if (pi1==0 || pi2==0) return; //Calculate the link points of the frames double x11 = pi1->xPos(); double y11 = pi1->yPos(); double x12 = x11+pi1->width(); double y12 = y11+pi1->height(); double x1mid = x11+(x12-x11)/2; double y1mid = y11+(y12-y11)/2; if (pi1->rotation()!=0.000) { FPoint tempPoint(0,0, x11, y11, pi1->rotation(), 1, 1); x11=tempPoint.x(); y11=tempPoint.y(); FPoint tempPoint2(0,0, x12, y12, pi1->rotation(), 1, 1); x12=tempPoint2.x(); y12=tempPoint2.y(); FPoint tempPoint3(0,0, x1mid, y1mid, pi1->rotation(), 1, 1); x1mid=tempPoint3.x(); y1mid=tempPoint3.y(); } double a1, b1, a2, b2; a1 = a2 = b1 = b2 = 0; double x21 = pi2->xPos(); double y21 = pi2->yPos(); double x22 = x21+pi2->width(); double y22 = y21+pi2->height(); double x2mid = x21 + pi2->width()/2; double y2mid = y21 + pi2->height()/2; if (pi2->rotation()!=0.000) { FPoint tempPoint(0,0, x21, y21, pi2->rotation(), 1, 1); x21=tempPoint.x(); y21=tempPoint.y(); FPoint tempPoint2(0,0, x22, y22, pi2->rotation(), 1, 1); x22=tempPoint2.x(); y22=tempPoint2.y(); FPoint tempPoint3(0,0, x2mid, y2mid, pi2->rotation(), 1, 1); x2mid=tempPoint3.x(); y2mid=tempPoint3.y(); } if (x22x12) { a1 = x12; a2 = x21; } if (y22y12) { b1 = y12; b2 = y21; } if (x21x11) { a1 = x1mid; a2 = x2mid; } if (x21x11) { a1 = x1mid; a2 = x2mid; } if (y21y11) { b1 = y1mid; b2 = y2mid; } if (y21y11) { b1 = y1mid; b2 = y2mid; } //When our points (in pt) are exactly the same, cover this too. #3634 if (x11==x21) { a1 = x1mid; a2 = x2mid; } if (y11==y21) { b1 = y1mid; b2 = y2mid; } //Set the link frame lines' endpoints start.setXY(a1-pi1->xPos(), b1-pi1->yPos()); start.transform(pi1->xPos(), pi1->yPos(), pi1->rotation(), 1, 1, false); end.setXY(a2-pi2->xPos(), b2-pi2->yPos()); end.transform(pi2->xPos(), pi2->yPos(), pi2->rotation(), 1, 1, false); } void Canvas::displayXYHUD(QPoint m) { if (!PrefsManager::instance()->appPrefs.showMouseCoordinates) return; double gx, gy, gh, gw, r; if (m_doc->m_Selection->isMultipleSelection()) { m_doc->m_Selection->setGroupRect(); m_doc->m_Selection->getGroupRect(&gx, &gy, &gw, &gh); r = 0.0; } else { PageItem* currItem = m_doc->m_Selection->itemAt(0); gx = currItem->xPos(); gy = currItem->yPos(); gw = currItem->width(); gh = currItem->height(); r = currItem->rotation(); } QMatrix ma; ma.translate(gx, gy); ma.rotate(r); FPoint n; if (m_doc->RotMode == 0) n = FPoint(0.0, 0.0); else if (m_doc->RotMode == 1) n = FPoint(gw, 0.0); else if (m_doc->RotMode == 2) n = FPoint(gw / 2.0, gh / 2.0); else if (m_doc->RotMode == 3) n = FPoint(0.0, gh); else if (m_doc->RotMode == 4) n = FPoint(gw, gh); gx = ma.m11() * n.x() + ma.m21() * n.y() + ma.dx(); gy = ma.m22() * n.y() + ma.m12() * n.x() + ma.dy(); if (m_doc->guidesSettings.rulerMode) { gx -= m_doc->currentPage()->xOffset(); gy -= m_doc->currentPage()->yOffset(); } gx -= m_doc->rulerXoffset; gy -= m_doc->rulerYoffset; displayXYHUD(m, gx, gy); } void Canvas::displayCorrectedXYHUD(QPoint m, double x, double y) { if (!PrefsManager::instance()->appPrefs.showMouseCoordinates) return; double gx = x; double gy = y; if (m_doc->guidesSettings.rulerMode) { gx -= m_doc->currentPage()->xOffset(); gy -= m_doc->currentPage()->yOffset(); } gx -= m_doc->rulerXoffset; gy -= m_doc->rulerYoffset; QToolTip::showText(m + QPoint(5, 5), tr("X: %1\nY: %2").arg(value2String(gx, m_doc->unitIndex(), true, true)).arg(value2String(gy, m_doc->unitIndex(), true, true)), this); } void Canvas::displayCorrectedSingleHUD(QPoint m, double val, bool isX) { if (!PrefsManager::instance()->appPrefs.showMouseCoordinates) return; double gx = val; if (isX) { if (m_doc->guidesSettings.rulerMode) gx -= m_doc->currentPage()->xOffset(); gx -= m_doc->rulerXoffset; QToolTip::showText(m + QPoint(5, 5), tr("X: %1").arg(value2String(gx, m_doc->unitIndex(), true, true)), this); } else { if (m_doc->guidesSettings.rulerMode) gx -= m_doc->currentPage()->yOffset(); gx -= m_doc->rulerYoffset; QToolTip::showText(m + QPoint(5, 5), tr("Y: %1").arg(value2String(gx, m_doc->unitIndex(), true, true)), this); } } void Canvas::displayXYHUD(QPoint m, double x, double y) { if (!PrefsManager::instance()->appPrefs.showMouseCoordinates) return; QToolTip::showText(m + QPoint(5, 5), tr("X: %1\nY: %2").arg(value2String(x, m_doc->unitIndex(), true, true)).arg(value2String(y, m_doc->unitIndex(), true, true)), this); } void Canvas::displaySizeHUD(QPoint m, double x, double y, bool isLine) { if (!PrefsManager::instance()->appPrefs.showMouseCoordinates) return; if (isLine) QToolTip::showText(m + QPoint(5, 5), tr("Length: %1\nAngle: %2").arg(value2String(x, m_doc->unitIndex(), true, true)).arg(value2String(y, SC_DEGREES, true, true)), this); else QToolTip::showText(m + QPoint(5, 5), tr("Width: %1\nHeight: %2").arg(value2String(x, m_doc->unitIndex(), true, true)).arg(value2String(y, m_doc->unitIndex(), true, true)), this); } void Canvas::displayRotHUD(QPoint m, double rot) { if (!PrefsManager::instance()->appPrefs.showMouseCoordinates) return; double r; if (rot < 0.0) r = rot * -1.0; else r = 360.0 - rot; QToolTip::showText(m + QPoint(5, 5), tr("Angle: %1").arg(value2String(r, SC_DEGREES, true, true)), this); } void Canvas::setupEditHRuler(PageItem * item, bool forceAndReset) { static QString rulerItemRef; static double rulerDumbHash(0.0); if( (rulerItemRef != item->itemName()) || forceAndReset ) { rulerItemRef = item->itemName(); rulerDumbHash = 0.0; } double controlHash(0.0); controlHash = item->xPos() + item->yPos() * 1.0 + item->ColGap * 2.0 + item->Cols * 3.0 + item->textToFrameDistLeft() * 4.0 + item->textToFrameDistRight() * 5.0 + item->currentStyle().firstIndent() * 6.0 + item->currentStyle().leftMargin() * 7.0 + item->width() * 8.0 + item->currentStyle().rightMargin() * 9.0 + (item->imageFlippedH() ? 32.32 : 13.13); foreach(const ParagraphStyle::TabRecord& tabrec, item->currentStyle().tabValues()) { controlHash += tabrec.tabPosition; } // qDebug()<<"Canvas::setupEditHRuler"<horizRuler->setItem(item); m_view->horizRuler->update(); }