doc/src/examples/basicdrawing.qdoc
changeset 0 1918ee327afb
equal deleted inserted replaced
-1:000000000000 0:1918ee327afb
       
     1 /****************************************************************************
       
     2 **
       
     3 ** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies).
       
     4 ** All rights reserved.
       
     5 ** Contact: Nokia Corporation (qt-info@nokia.com)
       
     6 **
       
     7 ** This file is part of the documentation of the Qt Toolkit.
       
     8 **
       
     9 ** $QT_BEGIN_LICENSE:LGPL$
       
    10 ** No Commercial Usage
       
    11 ** This file contains pre-release code and may not be distributed.
       
    12 ** You may use this file in accordance with the terms and conditions
       
    13 ** contained in the Technology Preview License Agreement accompanying
       
    14 ** this package.
       
    15 **
       
    16 ** GNU Lesser General Public License Usage
       
    17 ** Alternatively, this file may be used under the terms of the GNU Lesser
       
    18 ** General Public License version 2.1 as published by the Free Software
       
    19 ** Foundation and appearing in the file LICENSE.LGPL included in the
       
    20 ** packaging of this file.  Please review the following information to
       
    21 ** ensure the GNU Lesser General Public License version 2.1 requirements
       
    22 ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
       
    23 **
       
    24 ** In addition, as a special exception, Nokia gives you certain additional
       
    25 ** rights.  These rights are described in the Nokia Qt LGPL Exception
       
    26 ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
       
    27 **
       
    28 ** If you have questions regarding the use of this file, please contact
       
    29 ** Nokia at qt-info@nokia.com.
       
    30 **
       
    31 **
       
    32 **
       
    33 **
       
    34 **
       
    35 **
       
    36 **
       
    37 **
       
    38 ** $QT_END_LICENSE$
       
    39 **
       
    40 ****************************************************************************/
       
    41 
       
    42 /*!
       
    43     \example painting/basicdrawing
       
    44     \title Basic Drawing Example
       
    45 
       
    46     The Basic Drawing example shows how to display basic graphics
       
    47     primitives in a variety of styles using the QPainter class.
       
    48 
       
    49     QPainter performs low-level painting on widgets and other paint
       
    50     devices. The class can draw everything from simple lines to
       
    51     complex shapes like pies and chords. It can also draw aligned text
       
    52     and pixmaps. Normally, it draws in a "natural" coordinate system,
       
    53     but it can in addition do view and world transformation.
       
    54 
       
    55     \image basicdrawing-example.png
       
    56 
       
    57     The example provides a render area, displaying the currently
       
    58     active shape, and lets the user manipulate the rendered shape and
       
    59     its appearance using the QPainter parameters: The user can change
       
    60     the active shape (\gui Shape), and modify the QPainter's pen (\gui
       
    61     {Pen Width}, \gui {Pen Style}, \gui {Pen Cap}, \gui {Pen Join}),
       
    62     brush (\gui {Brush Style}) and render hints (\gui
       
    63     Antialiasing). In addition the user can rotate a shape (\gui
       
    64     Transformations); behind the scenes we use QPainter's ability to
       
    65     manipulate the coordinate system to perform the rotation.
       
    66 
       
    67     The Basic Drawing example consists of two classes:
       
    68 
       
    69     \list
       
    70     \o \c RenderArea is a custom widget that renders multiple
       
    71        copies of the currently active shape.
       
    72     \o \c Window is the application's main window displaying a
       
    73        \c RenderArea widget in addition to several parameter widgets.
       
    74     \endlist
       
    75 
       
    76     First we will review the \c Window class, then we will take a
       
    77     look at the \c RenderArea class.
       
    78 
       
    79     \section1 Window Class Definition
       
    80 
       
    81     The Window class inherits QWidget, and is the application's main
       
    82     window displaying a \c RenderArea widget in addition to several
       
    83     parameter widgets.
       
    84 
       
    85     \snippet examples/painting/basicdrawing/window.h 0
       
    86 
       
    87     We declare the various widgets, and three private slots updating
       
    88     the \c RenderArea widget: The \c shapeChanged() slot updates the
       
    89     \c RenderArea widget when the user changes the currently active
       
    90     shape. We call the \c penChanged() slot when either of the
       
    91     QPainter's pen parameters changes. And the \c brushChanged() slot
       
    92     updates the \c RenderArea widget when the user changes the
       
    93     painter's brush style.
       
    94 
       
    95     \section1 Window Class Implementation
       
    96 
       
    97     In the constructor we create and initialize the various widgets
       
    98     appearing in the main application window.
       
    99 
       
   100     \snippet examples/painting/basicdrawing/window.cpp 1
       
   101 
       
   102     First we create the \c RenderArea widget that will render the
       
   103     currently active shape. Then we create the \gui Shape combobox,
       
   104     and add the associated items (i.e. the different shapes a QPainter
       
   105     can draw).
       
   106 
       
   107     \snippet examples/painting/basicdrawing/window.cpp 2
       
   108 
       
   109     QPainter's pen is a QPen object; the QPen class defines how a
       
   110     painter should draw lines and outlines of shapes. A pen has
       
   111     several properties: Width, style, cap and join.
       
   112 
       
   113     A pen's width can be \e zero or greater, but the most common width
       
   114     is zero. Note that this doesn't mean 0 pixels, but implies that
       
   115     the shape is drawn as smoothly as possible although perhaps not
       
   116     mathematically correct.
       
   117 
       
   118     We create a QSpinBox for the \gui {Pen Width} parameter.
       
   119 
       
   120     \snippet examples/painting/basicdrawing/window.cpp 3
       
   121 
       
   122     The pen style defines the line type. The default style is solid
       
   123     (Qt::SolidLine). Setting the style to none (Qt::NoPen) tells the
       
   124     painter to not draw lines or outlines.  The pen cap defines how
       
   125     the end points of lines are drawn.  And the pen join defines how
       
   126     two lines join when multiple connected lines are drawn. The cap
       
   127     and join only apply to lines with a width of 1 pixel or greater.
       
   128 
       
   129     We create \l {QComboBox}es for each of the \gui {Pen Style}, \gui
       
   130     {Pen Cap} and \gui {Pen Join} parameters, and adds the associated
       
   131     items (i.e the values of the Qt::PenStyle, Qt::PenCapStyle and
       
   132     Qt::PenJoinStyle enums respectively).
       
   133 
       
   134     \snippet examples/painting/basicdrawing/window.cpp 4
       
   135 
       
   136     The QBrush class defines the fill pattern of shapes drawn by a
       
   137     QPainter. The default brush style is Qt::NoBrush. This style tells
       
   138     the painter to not fill shapes. The standard style for filling is
       
   139     Qt::SolidPattern.
       
   140 
       
   141     We create a QComboBox for the \gui {Brush Style} parameter, and add
       
   142     the associated items (i.e. the values of the Qt::BrushStyle enum).
       
   143 
       
   144     \snippet examples/painting/basicdrawing/window.cpp 5
       
   145     \snippet examples/painting/basicdrawing/window.cpp 6
       
   146 
       
   147     Antialiasing is a feature that "smoothes" the pixels to create
       
   148     more even and less jagged lines, and can be applied using
       
   149     QPainter's render hints. QPainter::RenderHints are used to specify
       
   150     flags to QPainter that may or may not be respected by any given
       
   151     engine.
       
   152 
       
   153     We simply create a QCheckBox for the \gui Antialiasing option.
       
   154 
       
   155     \snippet examples/painting/basicdrawing/window.cpp 7
       
   156 
       
   157     The \gui Transformations option implies a manipulation of the
       
   158     coordinate system that will appear as if the rendered shape is
       
   159     rotated in three dimensions.
       
   160 
       
   161     We use the QPainter::translate(), QPainter::rotate() and
       
   162     QPainter::scale() functions to implement this feature represented
       
   163     in the main application window by a simple QCheckBox.
       
   164 
       
   165     \snippet examples/painting/basicdrawing/window.cpp 8
       
   166 
       
   167     Then we connect the parameter widgets with their associated slots
       
   168     using the static QObject::connect() function, ensuring that the \c
       
   169     RenderArea widget is updated whenever the user changes the shape,
       
   170     or any of the other parameters.
       
   171 
       
   172     \snippet examples/painting/basicdrawing/window.cpp 9
       
   173     \snippet examples/painting/basicdrawing/window.cpp 10
       
   174 
       
   175     Finally, we add the various widgets to a layout, and call the \c
       
   176     shapeChanged(), \c penChanged(), and \c brushChanged() slots to
       
   177     initialize the application. We also turn on antialiasing.
       
   178 
       
   179     \snippet examples/painting/basicdrawing/window.cpp 11
       
   180 
       
   181     The \c shapeChanged() slot is called whenever the user changes the
       
   182     currently active shape.
       
   183 
       
   184     First we retrieve the shape the user has chosen using the
       
   185     QComboBox::itemData() function. This function returns the data for
       
   186     the given role in the given index in the combobox. We use
       
   187     QComboBox::currentIndex() to retrieve the index of the shape, and
       
   188     the role is defined by the Qt::ItemDataRole enum; \c IdRole is an
       
   189     alias for Qt::UserRole.
       
   190 
       
   191     Note that Qt::UserRole is only the first role that can be used for
       
   192     application-specific purposes. If you need to store different data
       
   193     in the same index, you can use different roles by simply
       
   194     incrementing the value of Qt::UserRole, for example: 'Qt::UserRole
       
   195     + 1' and 'Qt::UserRole + 2'. However, it is a good programming
       
   196     practice to give each role their own name: 'myFirstRole =
       
   197     Qt::UserRole + 1' and 'mySecondRole = Qt::UserRole + 2'. Even
       
   198     though we only need a single role in this particular example, we
       
   199     add the following line of code to the beginning of the \c
       
   200     window.cpp file.
       
   201 
       
   202     \snippet examples/painting/basicdrawing/window.cpp 0
       
   203 
       
   204     The QComboBox::itemData() function returns the data as a QVariant,
       
   205     so we need to cast the data to \c RenderArea::Shape. If there is
       
   206     no data for the given role, the function returns
       
   207     QVariant::Invalid.
       
   208 
       
   209     In the end we call the \c RenderArea::setShape() slot to update
       
   210     the \c RenderArea widget.
       
   211 
       
   212     \snippet examples/painting/basicdrawing/window.cpp 12
       
   213 
       
   214     We call the \c penChanged() slot whenever the user changes any of
       
   215     the pen parameters. Again we use the QComboBox::itemData()
       
   216     function to retrieve the parameters, and then we call the \c
       
   217     RenderArea::setPen() slot to update the \c RenderArea widget.
       
   218 
       
   219     \snippet examples/painting/basicdrawing/window.cpp 13
       
   220 
       
   221     The brushChanged() slot is called whenever the user changes the
       
   222     brush parameter which we retrieve using the QComboBox::itemData()
       
   223     function as before.
       
   224 
       
   225     \snippet examples/painting/basicdrawing/window.cpp 14
       
   226 
       
   227     If the brush parameter is a gradient fill, special actions are
       
   228     required.
       
   229 
       
   230     The QGradient class is used in combination with QBrush to specify
       
   231     gradient fills. Qt currently supports three types of gradient
       
   232     fills: linear, radial and conical. Each of these is represented by
       
   233     a subclass of QGradient: QLinearGradient, QRadialGradient and
       
   234     QConicalGradient.
       
   235 
       
   236     So if the brush style is Qt::LinearGradientPattern, we first
       
   237     create a QLinearGradient object with interpolation area between
       
   238     the coordinates passed as arguments to the constructor. The
       
   239     positions are specified using logical coordinates. Then we set the
       
   240     gradient's colors using the QGradient::setColorAt() function. The
       
   241     colors is defined using stop points which are composed by a
       
   242     position (between 0 and 1) and a QColor. The set of stop points
       
   243     describes how the gradient area should be filled. A gradient can
       
   244     have an arbitrary number of stop points.
       
   245 
       
   246     In the end we call \c RenderArea::setBrush() slot to update the \c
       
   247     RenderArea widget's brush with the QLinearGradient object.
       
   248 
       
   249     \snippet examples/painting/basicdrawing/window.cpp 15
       
   250 
       
   251     A similar pattern of actions, as the one used for QLinearGradient,
       
   252     is used in the cases of Qt::RadialGradientPattern and
       
   253     Qt::ConicalGradientPattern.
       
   254 
       
   255     The only difference is the arguments passed to the constructor:
       
   256     Regarding the QRadialGradient constructor the first argument is
       
   257     the center, and the second the radial gradient's radius. The third
       
   258     argument is optional, but can be used to define the focal point of
       
   259     the gradient inside the circle (the default focal point is the
       
   260     circle center).  Regarding the QConicalGradient constructor, the
       
   261     first argument specifies the center of the conical, and the second
       
   262     specifies the start angle of the interpolation.
       
   263 
       
   264     \snippet examples/painting/basicdrawing/window.cpp 16
       
   265 
       
   266     If the brush style is Qt::TexturePattern we create a QBrush from a
       
   267     QPixmap. Then we call \c RenderArea::setBrush() slot to update the
       
   268     \c RenderArea widget with the newly created brush.
       
   269 
       
   270     \snippet examples/painting/basicdrawing/window.cpp 17
       
   271 
       
   272     Otherwise we simply create a brush with the given style and a
       
   273     green color, and then call \c RenderArea::setBrush() slot to
       
   274     update the \c RenderArea widget with the newly created brush.
       
   275 
       
   276     \section1 RenderArea Class Definition
       
   277 
       
   278     The \c RenderArea class inherits QWidget, and renders multiple
       
   279     copies of the currently active shape using a QPainter.
       
   280 
       
   281     \snippet examples/painting/basicdrawing/renderarea.h 0
       
   282 
       
   283     First we define a public \c Shape enum to hold the different
       
   284     shapes that can be rendered by the widget (i.e the shapes that can
       
   285     be rendered by a QPainter). Then we reimplement the constructor as
       
   286     well as two of QWidget's public functions: \l
       
   287     {QWidget::minimumSizeHint()}{minimumSizeHint()} and \l
       
   288     {QWidget::sizeHint()}{sizeHint()}.
       
   289 
       
   290     We also reimplement the QWidget::paintEvent() function to be able
       
   291     to draw the currently active shape according to the specified
       
   292     parameters.
       
   293 
       
   294     We declare several private slots: The \c setShape() slot changes
       
   295     the \c RenderArea's shape, the \c setPen() and \c setBrush() slots
       
   296     modify the widget's pen and brush, and the \c setAntialiased() and
       
   297     \c setTransformed() slots modify the widget's respective
       
   298     properties.
       
   299 
       
   300     \section1 RenderArea Class Implementation
       
   301 
       
   302     In the constructor we initialize some of the widget's variables.
       
   303 
       
   304     \snippet examples/painting/basicdrawing/renderarea.cpp 0
       
   305 
       
   306     We set its shape to be a \gui Polygon, its antialiased property to
       
   307     be false and we load an image into the widget's pixmap
       
   308     variable. In the end we set the widget's background role, defining
       
   309     the brush from the widget's \l {QWidget::palette}{palette} that
       
   310     will be used to render the background. QPalette::Base is typically
       
   311     white.
       
   312 
       
   313     \snippet examples/painting/basicdrawing/renderarea.cpp 2
       
   314 
       
   315     The \c RenderArea inherits QWidget's \l
       
   316     {QWidget::sizeHint()}{sizeHint} property holding the recommended
       
   317     size for the widget. If the value of this property is an invalid
       
   318     size, no size is recommended.
       
   319 
       
   320     The default implementation of the QWidget::sizeHint() function
       
   321     returns an invalid size if there is no layout for the widget, and
       
   322     returns the layout's preferred size otherwise.
       
   323 
       
   324     Our reimplementation of the function returns a QSize with a 400
       
   325     pixels width and a 200 pixels height.
       
   326 
       
   327     \snippet examples/painting/basicdrawing/renderarea.cpp 1
       
   328 
       
   329     \c RenderArea also inherits QWidget's
       
   330     \l{QWidget::minimumSizeHint()}{minimumSizeHint} property holding
       
   331     the recommended minimum size for the widget. Again, if the value
       
   332     of this property is an invalid size, no size is recommended.
       
   333 
       
   334     The default implementation of QWidget::minimumSizeHint() returns
       
   335     an invalid size if there is no layout for the widget, and returns
       
   336     the layout's minimum size otherwise.
       
   337 
       
   338     Our reimplementation of the function returns a QSize with a 100
       
   339     pixels width and a 100 pixels height.
       
   340 
       
   341     \snippet examples/painting/basicdrawing/renderarea.cpp 3
       
   342     \codeline
       
   343     \snippet examples/painting/basicdrawing/renderarea.cpp 4
       
   344     \codeline
       
   345     \snippet examples/painting/basicdrawing/renderarea.cpp 5
       
   346 
       
   347     The public \c setShape(), \c setPen() and \c setBrush() slots are
       
   348     called whenever we want to modify a \c RenderArea widget's shape,
       
   349     pen or brush. We set the shape, pen or brush according to the
       
   350     slot parameter, and call QWidget::update() to make the changes
       
   351     visible in the \c RenderArea widget.
       
   352 
       
   353     The QWidget::update() slot does not cause an immediate
       
   354     repaint; instead it schedules a paint event for processing when Qt
       
   355     returns to the main event loop.
       
   356 
       
   357     \snippet examples/painting/basicdrawing/renderarea.cpp 6
       
   358     \codeline
       
   359     \snippet examples/painting/basicdrawing/renderarea.cpp 7
       
   360 
       
   361     With the \c setAntialiased() and \c setTransformed() slots we
       
   362     change the state of the properties according to the slot
       
   363     parameter, and call the QWidget::update() slot to make the changes
       
   364     visible in the \c RenderArea widget.
       
   365 
       
   366     \snippet examples/painting/basicdrawing/renderarea.cpp 8
       
   367 
       
   368     Then we reimplement the QWidget::paintEvent() function. The first
       
   369     thing we do is to create the graphical objects we will need to
       
   370     draw the various shapes.
       
   371 
       
   372     We create a vector of four \l {QPoint}s. We use this vector to
       
   373     render the \gui Points, \gui Polyline and \gui Polygon
       
   374     shapes. Then we create a QRect, defining a rectangle in the plane,
       
   375     which we use as the bounding rectangle for all the shapes excluding
       
   376     the \gui Path and the \gui Pixmap.
       
   377 
       
   378     We also create a QPainterPath. The QPainterPath class provides a
       
   379     container for painting operations, enabling graphical shapes to be
       
   380     constructed and reused. A painter path is an object composed of a
       
   381     number of graphical building blocks, such as rectangles, ellipses,
       
   382     lines, and curves. For more information about the QPainterPath
       
   383     class, see the \l {painting/painterpaths}{Painter Paths}
       
   384     example. In this example, we create a painter path composed of one
       
   385     straight line and a Bezier curve.
       
   386 
       
   387     In addition we define a start angle and an arc length that we will
       
   388     use when drawing the \gui Arc, \gui Chord and \gui Pie shapes.
       
   389 
       
   390     \snippet examples/painting/basicdrawing/renderarea.cpp 9
       
   391 
       
   392     We create a QPainter for the \c RenderArea widget, and set the
       
   393     painters pen and brush according to the \c RenderArea's pen and
       
   394     brush. If the \gui Antialiasing parameter option is checked, we
       
   395     also set the painter's render hints. QPainter::Antialiasing
       
   396     indicates that the engine should antialias edges of primitives if
       
   397     possible.
       
   398 
       
   399     \snippet examples/painting/basicdrawing/renderarea.cpp 10
       
   400 
       
   401     Finally, we render the multiple copies of the \c RenderArea's
       
   402     shape. The number of copies is depending on the size of the \c
       
   403     RenderArea widget, and we calculate their positions using two \c
       
   404     for loops and the widgets height and width.
       
   405 
       
   406     For each copy we first save the current painter state (pushes the
       
   407     state onto a stack). Then we translate the coordinate system,
       
   408     using the QPainter::translate() function, to the position
       
   409     determined by the variables of the \c for loops. If we omit this
       
   410     translation of the coordinate system all the copies of the shape
       
   411     will be rendered on top of each other in the top left cormer of
       
   412     the \c RenderArea widget.
       
   413 
       
   414     \snippet examples/painting/basicdrawing/renderarea.cpp 11
       
   415 
       
   416     If the \gui Transformations parameter option is checked, we do an
       
   417     additional translation of the coordinate system before we rotate
       
   418     the coordinate system 60 degrees clockwise using the
       
   419     QPainter::rotate() function and scale it down in size using the
       
   420     QPainter::scale() function. In the end we translate the coordinate
       
   421     system back to where it was before we rotated and scaled it.
       
   422 
       
   423     Now, when rendering the shape, it will appear as if it was rotated
       
   424     in three dimensions.
       
   425 
       
   426     \snippet examples/painting/basicdrawing/renderarea.cpp 12
       
   427 
       
   428     Next, we identify the \c RenderArea's shape, and render it using
       
   429     the associated QPainter drawing function:
       
   430 
       
   431     \list
       
   432         \o QPainter::drawLine(),
       
   433         \o QPainter::drawPoints(),
       
   434         \o QPainter::drawPolyline(),
       
   435         \o QPainter::drawPolygon(),
       
   436         \o QPainter::drawRect(),
       
   437         \o QPainter::drawRoundedRect(),
       
   438         \o QPainter::drawEllipse(),
       
   439         \o QPainter::drawArc(),
       
   440         \o QPainter::drawChord(),
       
   441         \o QPainter::drawPie(),
       
   442         \o QPainter::drawPath(),
       
   443         \o QPainter::drawText() or
       
   444         \o QPainter::drawPixmap()
       
   445     \endlist
       
   446 
       
   447     Before we started rendering, we saved the current painter state
       
   448     (pushes the state onto a stack). The rationale for this is that we
       
   449     calculate each shape copy's position relative to the same point in
       
   450     the coordinate system. When translating the coordinate system, we
       
   451     lose the knowledge of this point unless we save the current
       
   452     painter state \e before we start the translating process.
       
   453 
       
   454     \snippet examples/painting/basicdrawing/renderarea.cpp 13
       
   455 
       
   456     Then, when we are finished rendering a copy of the shape we can
       
   457     restore the original painter state, with its associated coordinate
       
   458     system, using the QPainter::restore() function. In this way we
       
   459     ensure that the next shape copy will be rendered in the correct
       
   460     position.
       
   461 
       
   462     We could translate the coordinate system back using
       
   463     QPainter::translate() instead of saving the painter state. But
       
   464     since we in addition to translating the coordinate system (when
       
   465     the \gui Transformation parameter option is checked) both rotate
       
   466     and scale the coordinate system, the easiest solution is to save
       
   467     the current painter state.
       
   468 */