Newer
Older
/* -*- mode: C++ ; c-file-style: "stroustrup" -*- *****************************
* Qwt Widget Library
* Copyright (C) 1997 Josef Wilgen
* Copyright (C) 2002 Uwe Rathmann
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the Qwt License, Version 1.0
*****************************************************************************/
#include "qwt_picker.h"
#include "qwt_picker_machine.h"
#include "qwt_painter.h"
#include "qwt_math.h"
#include "qwt_widget_overlay.h"
#include <qapplication.h>
#include <qevent.h>
#include <qpainter.h>
#include <qframe.h>
#include <qcursor.h>
#include <qbitmap.h>
#include <qpointer.h>
#include <qpaintengine.h>
static inline QRegion qwtMaskRegion( const QRect &r, int penWidth )
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
const int pw = qMax( penWidth, 1 );
const int pw2 = penWidth / 2;
int x1 = r.left() - pw2;
int x2 = r.right() + 1 + pw2 + ( pw % 2 );
int y1 = r.top() - pw2;
int y2 = r.bottom() + 1 + pw2 + ( pw % 2 );
QRegion region;
region += QRect( x1, y1, x2 - x1, pw );
region += QRect( x1, y1, pw, y2 - y1 );
region += QRect( x1, y2 - pw, x2 - x1, pw );
region += QRect( x2 - pw, y1, pw, y2 - y1 );
return region;
}
static inline QRegion qwtMaskRegion( const QLine &l, int penWidth )
{
const int pw = qMax( penWidth, 1 );
const int pw2 = penWidth / 2;
QRegion region;
if ( l.x1() == l.x2() )
{
region += QRect( l.x1() - pw2, l.y1(),
pw, l.y2() ).normalized();
}
else if ( l.y1() == l.y2() )
{
region += QRect( l.x1(), l.y1() - pw2,
l.x2(), pw ).normalized();
}
class QwtPickerRubberband: public QwtWidgetOverlay
{
public:
QwtPickerRubberband( QwtPicker *, QWidget * );
virtual void drawOverlay( QPainter * ) const;
virtual QRegion maskHint() const;
class QwtPickerTracker: public QwtWidgetOverlay
{
public:
QwtPickerTracker( QwtPicker *, QWidget * );
protected:
virtual void drawOverlay( QPainter * ) const;
virtual QRegion maskHint() const;
QwtPicker *d_picker;
};
PrivateData():
enabled( false ),
stateMachine( NULL ),
resizeMode( QwtPicker::Stretch ),
rubberBand( QwtPicker::NoRubberBand ),
trackerMode( QwtPicker::AlwaysOff ),
isActive( false ),
trackerPosition( -1, -1 ),
mouseTracking( false ),
openGL( false )
{
}
bool enabled;
QwtPickerMachine *stateMachine;
QwtPicker::ResizeMode resizeMode;
QwtPicker::RubberBand rubberBand;
QPen rubberBandPen;
QwtPicker::DisplayMode trackerMode;
QPen trackerPen;
QFont trackerFont;
bool isActive;
QPoint trackerPosition;
bool mouseTracking; // used to save previous value
QPointer< QwtPickerRubberband > rubberBandOverlay;
QPointer< QwtPickerTracker> trackerOverlay;
bool openGL;
QwtPickerRubberband::QwtPickerRubberband(
QwtPicker *picker, QWidget *parent ):
QwtWidgetOverlay( parent ),
d_picker( picker )
void QwtPickerRubberband::drawOverlay( QPainter *painter ) const
{
painter->setPen( d_picker->rubberBandPen() );
d_picker->drawRubberBand( painter );
QwtPickerTracker::QwtPickerTracker(
QwtPicker *picker, QWidget *parent ):
QwtWidgetOverlay( parent ),
d_picker( picker )
QRegion QwtPickerTracker::maskHint() const
{
return d_picker->trackerRect( font() );
}
void QwtPickerTracker::drawOverlay( QPainter *painter ) const
{
painter->setPen( d_picker->trackerPen() );
d_picker->drawTracker( painter );
Creates an picker that is enabled, but without a state machine.
rubber band and tracker are disabled.
QwtPicker::QwtPicker( QWidget *parent ):
QObject( parent )
\param trackerMode Tracker mode
\param parent Parent widget, that will be observed
*/
QwtPicker::QwtPicker( RubberBand rubberBand,
DisplayMode trackerMode, QWidget *parent ):
QObject( parent )
delete d_data->rubberBandOverlay;
delete d_data->trackerOverlay;
//! Initialize the picker - used by the constructors
void QwtPicker::init( QWidget *parent,
RubberBand rubberBand, DisplayMode trackerMode )
{
d_data = new PrivateData;
d_data->rubberBand = rubberBand;
d_data->trackerFont = parent->font();
d_data->mouseTracking = parent->hasMouseTracking();
Set a state machine and delete the previous one
\param stateMachine State machine
\sa stateMachine()
void QwtPicker::setStateMachine( QwtPickerMachine *stateMachine )
reset();
delete d_data->stateMachine;
d_data->stateMachine = stateMachine;
if ( d_data->stateMachine )
d_data->stateMachine->reset();
}
}
/*!
return d_data->stateMachine;
}
/*!
\return Assigned state machine
\sa setStateMachine()
*/
const QwtPickerMachine *QwtPicker::stateMachine() const
{
return d_data->stateMachine;
}
//! Return the parent widget, where the selection happens
QWidget *QwtPicker::parentWidget()
{
QObject *obj = parent();
if ( obj && obj->isWidgetType() )
return NULL;
}
//! Return the parent widget, where the selection happens
const QWidget *QwtPicker::parentWidget() const
{
QObject *obj = parent();
if ( obj && obj->isWidgetType() )
The default value is NoRubberBand.
\sa rubberBand(), RubberBand, setRubberBandPen()
*/
\sa setRubberBand(), RubberBand, rubberBandPen()
*/
QwtPicker::RubberBand QwtPicker::rubberBand() const
{
return d_data->rubberBand;
}
/*!
\brief Set the display mode of the tracker.
A tracker displays information about current position of
the cursor as a string. The display mode controls
if the tracker has to be displayed whenever the observed
widget has focus and cursor (AlwaysOn), never (AlwaysOff), or
only when the selection is active (ActiveOnly).
\param mode Tracker display mode
\warning In case of AlwaysOn, mouseTracking will be enabled
for the observed widget.
\sa trackerMode(), DisplayMode
*/
/*!
\return Tracker display mode
\sa setTrackerMode(), DisplayMode
*/
QwtPicker::DisplayMode QwtPicker::trackerMode() const
/*!
\brief Set the resize mode.
The resize mode controls what to do with the selected points of an active
selection when the observed widget is resized.
Stretch means the points are scaled according to the new
size, KeepSize means the points remain unchanged.
The default mode is Stretch.
\param mode Resize mode
\sa resizeMode(), ResizeMode
*/
/*!
\return Resize mode
\sa setResizeMode(), ResizeMode
*/
QwtPicker::ResizeMode QwtPicker::resizeMode() const
return d_data->resizeMode;
}
/*!
\brief En/disable the picker
When enabled is true an event filter is installed for
the observed widget, otherwise the event filter is removed.
\param enabled true or false
\sa isEnabled(), eventFilter()
*/
d_data->enabled = enabled;
QWidget *w = parentWidget();
}
updateDisplay();
}
}
/*!
\return true when enabled, false otherwise
*/
bool QwtPicker::isEnabled() const
{
return d_data->enabled;
}
/*!
Set the font for the tracker
\param font Tracker font
\sa trackerFont(), setTrackerMode(), setTrackerPen()
*/
d_data->trackerFont = font;
updateDisplay();
}
}
/*!
\return Tracker font
\sa setTrackerFont(), trackerMode(), trackerPen()
*/
QFont QwtPicker::trackerFont() const
{
return d_data->trackerFont;
}
/*!
Set the pen for the tracker
\param pen Tracker pen
\sa trackerPen(), setTrackerMode(), setTrackerFont()
*/
d_data->trackerPen = pen;
updateDisplay();
}
}
/*!
\return Tracker pen
\sa setTrackerPen(), trackerMode(), trackerFont()
*/
QPen QwtPicker::trackerPen() const
{
return d_data->trackerPen;
}
/*!
Set the pen for the rubberband
d_data->rubberBandPen = pen;
updateDisplay();
}
}
/*!
\sa setRubberBandPen(), rubberBand()
*/
QPen QwtPicker::rubberBandPen() const
{
return d_data->rubberBandPen;
}
/*!
\brief Return the label for a position
In case of HLineRubberBand the label is the value of the
y position, in case of VLineRubberBand the value of the x position.
Otherwise the label contains x and y position separated by a ',' .
The format for the string conversion is "%d".
\param pos Position
\return Converted position as string
*/
switch ( rubberBand() )
{
case HLineRubberBand:
label.sprintf( "%d", pos.y() );
break;
case VLineRubberBand:
label.sprintf( "%d", pos.x() );
break;
default:
label.sprintf( "%d, %d", pos.x(), pos.y() );
\return Region for the mask
\sa QWidget::setMask()
*/
QRegion QwtPicker::rubberBandMask() const
if ( !isActive() || rubberBand() == NoRubberBand ||
rubberBandPen().style() == Qt::NoPen )
{
return mask;
QwtPickerMachine::SelectionType selectionType =
QwtPickerMachine::NoSelection;
if ( d_data->stateMachine )
selectionType = d_data->stateMachine->selectionType();
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
switch ( selectionType )
{
case QwtPickerMachine::NoSelection:
case QwtPickerMachine::PointSelection:
{
if ( pa.count() < 1 )
return mask;
const QPoint pos = pa[0];
const int pw = rubberBandPen().width();
const QRect pRect = pickArea().boundingRect().toRect();
switch ( rubberBand() )
{
case VLineRubberBand:
{
mask += qwtMaskRegion( QLine( pos.x(), pRect.top(),
pos.x(), pRect.bottom() ), pw );
break;
}
case HLineRubberBand:
{
mask += qwtMaskRegion( QLine( pRect.left(), pos.y(),
pRect.right(), pos.y() ), pw );
break;
}
case CrossRubberBand:
{
mask += qwtMaskRegion( QLine( pos.x(), pRect.top(),
pos.x(), pRect.bottom() ), pw );
mask += qwtMaskRegion( QLine( pRect.left(), pos.y(),
pRect.right(), pos.y() ), pw );
break;
}
default:
break;
}
}
case QwtPickerMachine::RectSelection:
{
if ( pa.count() < 2 )
return mask;
const int pw = rubberBandPen().width();
switch ( rubberBand() )
{
case RectRubberBand:
{
const QRect r = QRect( pa.first(), pa.last() );
mask = qwtMaskRegion( r.normalized(), pw );
break;
}
case EllipseRubberBand:
{
const QRect r = QRect( pa.first(), pa.last() );
mask += r.adjusted( -pw, -pw, pw, pw );
break;
}
default:
break;
}
}
case QwtPickerMachine::PolygonSelection:
{
const int pw = rubberBandPen().width();
if ( pw <= 1 )
{
// because of the join style we better
// return a mask for a pen width <= 1 only
const int off = 2 * pw;
const QRect r = pa.boundingRect();
mask += r.adjusted( -off, -off, off, off );
}
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
return mask;
}
/*!
Draw a rubber band, depending on rubberBand()
\param painter Painter, initialized with a clip region
\sa rubberBand(), RubberBand
*/
void QwtPicker::drawRubberBand( QPainter *painter ) const
{
if ( !isActive() || rubberBand() == NoRubberBand ||
rubberBandPen().style() == Qt::NoPen )
{
return;
}
const QPolygon pa = adjustedPoints( d_data->pickedPoints );
QwtPickerMachine::SelectionType selectionType =
QwtPickerMachine::NoSelection;
if ( d_data->stateMachine )
selectionType = d_data->stateMachine->selectionType();
switch ( selectionType )
{
case QwtPickerMachine::NoSelection:
case QwtPickerMachine::PointSelection:
{
if ( pa.count() < 1 )
return;
const QPoint pos = pa[0];
const QRect pRect = pickArea().boundingRect().toRect();
switch ( rubberBand() )
{
case VLineRubberBand:
{
QwtPainter::drawLine( painter, pos.x(),
pRect.top(), pos.x(), pRect.bottom() );
break;
}
case HLineRubberBand:
{
QwtPainter::drawLine( painter, pRect.left(),
pos.y(), pRect.right(), pos.y() );
break;
}
case CrossRubberBand:
{
QwtPainter::drawLine( painter, pos.x(),
pRect.top(), pos.x(), pRect.bottom() );
QwtPainter::drawLine( painter, pRect.left(),
pos.y(), pRect.right(), pos.y() );
break;
}
default:
break;
}
break;
case QwtPickerMachine::RectSelection:
{
if ( pa.count() < 2 )
return;
const QRect rect = QRect( pa.first(), pa.last() ).normalized();
switch ( rubberBand() )
{
case EllipseRubberBand:
{
QwtPainter::drawEllipse( painter, rect );
break;
}
case RectRubberBand:
{
QwtPainter::drawRect( painter, rect );
break;
}
default:
break;
}
}
case QwtPickerMachine::PolygonSelection:
{
if ( rubberBand() == PolygonRubberBand )
painter->drawPolyline( pa );
}
}
/*!
Draw the tracker
\param painter Painter
\sa trackerRect(), trackerText()
*/
void QwtPicker::drawTracker( QPainter *painter ) const
{
const QRect textRect = trackerRect( painter->font() );
if ( !textRect.isEmpty() )
{
const QwtText label = trackerText( d_data->trackerPosition );
if ( !label.isEmpty() )
label.draw( painter, textRect );
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
/*!
\brief Map the pickedPoints() into a selection()
adjustedPoints() maps the points, that have been collected on
the parentWidget() into a selection(). The default implementation
simply returns the points unmodified.
The reason, why a selection() differs from the picked points
depends on the application requirements. F.e. :
- A rectangular selection might need to have a specific aspect ratio only.\n
- A selection could accept non intersecting polygons only.\n
- ...\n
The example below is for a rectangular selection, where the first
point is the center of the selected rectangle.
\par Example
\verbatim QPolygon MyPicker::adjustedPoints(const QPolygon &points) const
{
QPolygon adjusted;
if ( points.size() == 2 )
{
const int width = qAbs(points[1].x() - points[0].x());
const int height = qAbs(points[1].y() - points[0].y());
QRect rect(0, 0, 2 * width, 2 * height);
rect.moveCenter(points[0]);
adjusted += rect.topLeft();
adjusted += rect.bottomRight();
}
return adjusted;
}\endverbatim\n
\param points Selected points
\return Selected points unmodified
*/
QPolygon QwtPicker::adjustedPoints( const QPolygon &points ) const
{
return points;
}
/*!
\return Selected points
\sa pickedPoints(), adjustedPoints()
*/
QPolygon QwtPicker::selection() const
{
return adjustedPoints( d_data->pickedPoints );
}
//! \return Current position of the tracker
QPoint QwtPicker::trackerPosition() const
/*!
Calculate the bounding rectangle for the tracker text
from the current position of the tracker
\param font Font of the tracker text
\return Bounding rectangle of the tracker text
\sa trackerPosition()
*/
QRect QwtPicker::trackerRect( const QFont &font ) const
if ( trackerMode() == AlwaysOff ||
return QRect();
}
if ( d_data->trackerPosition.x() < 0 || d_data->trackerPosition.y() < 0 )
return QRect();
const QSizeF textSize = text.textSize( font );
QRect textRect( 0, 0, qCeil( textSize.width() ), qCeil( textSize.height() ) );
const QPoint &pos = d_data->trackerPosition;
int alignment = 0;
if ( isActive() && d_data->pickedPoints.count() > 1
&& rubberBand() != NoRubberBand )
{
d_data->pickedPoints[int( d_data->pickedPoints.count() ) - 2];
alignment |= ( pos.x() >= last.x() ) ? Qt::AlignRight : Qt::AlignLeft;
alignment |= ( pos.y() > last.y() ) ? Qt::AlignBottom : Qt::AlignTop;
}
else
alignment = Qt::AlignTop | Qt::AlignRight;
const int margin = 5;
int x = pos.x();
if ( alignment & Qt::AlignLeft )
x -= textRect.width() + margin;
else if ( alignment & Qt::AlignRight )
x += margin;
int y = pos.y();
if ( alignment & Qt::AlignBottom )
y += margin;
else if ( alignment & Qt::AlignTop )
y -= textRect.height() + margin;
int right = qMin( textRect.right(), pickRect.right() - margin );
int bottom = qMin( textRect.bottom(), pickRect.bottom() - margin );
textRect.moveBottomRight( QPoint( right, bottom ) );
int left = qMax( textRect.left(), pickRect.left() + margin );
int top = qMax( textRect.top(), pickRect.top() + margin );
textRect.moveTopLeft( QPoint( left, top ) );
return textRect;
}
/*!
\brief Event filter
When isEnabled() is true all events of the observed widget are filtered.
Mouse and keyboard events are translated into widgetMouse- and widgetKey-
and widgetWheel-events. Paint and Resize events are handled to keep
rubber band and tracker up to date.
\param object Object to be filtered
\param event Event
\return Always false.
\sa widgetEnterEvent(), widgetLeaveEvent(),
widgetMousePressEvent(), widgetMouseReleaseEvent(),
widgetWheelEvent(), widgetKeyPressEvent(), widgetKeyReleaseEvent(),
QObject::installEventFilter(), QObject::event()
bool QwtPicker::eventFilter( QObject *object, QEvent *event )
if ( object && object == parentWidget() )
{
switch ( event->type() )
{
case QEvent::Resize:
{
const QResizeEvent *re = static_cast<QResizeEvent *>( event );
if ( d_data->resizeMode == Stretch )
stretchSelection( re->oldSize(), re->size() );
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
break;
}
case QEvent::Enter:
{
widgetEnterEvent( event );
break;
}
case QEvent::Leave:
{
widgetLeaveEvent( event );
break;
}
case QEvent::MouseButtonPress:
{
widgetMousePressEvent( static_cast<QMouseEvent *>( event ) );
break;
}
case QEvent::MouseButtonRelease:
{
widgetMouseReleaseEvent( static_cast<QMouseEvent *>( event ) );
break;
}
case QEvent::MouseButtonDblClick:
{
widgetMouseDoubleClickEvent( static_cast<QMouseEvent *>( event ) );
break;
}
case QEvent::MouseMove:
{
widgetMouseMoveEvent( static_cast<QMouseEvent *>( event ) );
break;
}
case QEvent::KeyPress:
{
widgetKeyPressEvent( static_cast<QKeyEvent *>( event ) );
break;
}
case QEvent::KeyRelease:
{
widgetKeyReleaseEvent( static_cast<QKeyEvent *>( event ) );
break;
}
case QEvent::Wheel:
{
widgetWheelEvent( static_cast<QWheelEvent *>( event ) );
break;
}
default:
break;
}
}
return false;
}
/*!
Handle a mouse press event for the observed widget.
\sa eventFilter(), widgetMouseReleaseEvent(),
widgetMouseDoubleClickEvent(), widgetMouseMoveEvent(),
widgetWheelEvent(), widgetKeyPressEvent(), widgetKeyReleaseEvent()
*/
void QwtPicker::widgetMousePressEvent( QMouseEvent *mouseEvent )
}
/*!
Handle a mouse move event for the observed widget.
\sa eventFilter(), widgetMousePressEvent(), widgetMouseReleaseEvent(),
widgetMouseDoubleClickEvent(),
widgetWheelEvent(), widgetKeyPressEvent(), widgetKeyReleaseEvent()
*/
void QwtPicker::widgetMouseMoveEvent( QMouseEvent *mouseEvent )
if ( pickArea().contains( mouseEvent->pos() ) )
d_data->trackerPosition = mouseEvent->pos();
transition( mouseEvent );
}
/*!
Handle a enter event for the observed widget.
\param event Qt event
\sa eventFilter(), widgetMousePressEvent(), widgetMouseReleaseEvent(),
widgetMouseDoubleClickEvent(),
widgetWheelEvent(), widgetKeyPressEvent(), widgetKeyReleaseEvent()