doc/src/examples/stardelegate.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 itemviews/stardelegate
       
    44     \title Star Delegate Example
       
    45 
       
    46     The Star Delegate example shows how to create a delegate that
       
    47     can paint itself and that supports editing.
       
    48 
       
    49     \image stardelegate.png The Star Delegate Example
       
    50 
       
    51     When displaying data in a QListView, QTableView, or QTreeView,
       
    52     the individual items are drawn by a
       
    53     \l{Delegate Classes}{delegate}. Also, when the user starts
       
    54     editing an item (e.g., by double-clicking the item), the delegate
       
    55     provides an editor widget that is placed on top of the item while
       
    56     editing takes place.
       
    57 
       
    58     Delegates are subclasses of QAbstractItemDelegate. Qt provides
       
    59     QItemDelegate, which inherits QAbstractItemDelegate and handles
       
    60     the most common data types (notably \c int and QString). If we
       
    61     need to support custom data types, or want to customize the
       
    62     rendering or the editing for existing data types, we can subclass
       
    63     QAbstractItemDelegate or QItemDelegate. See \l{Delegate Classes}
       
    64     for more information about delegates, and \l{Model/View
       
    65     Programming} if you need a high-level introduction to Qt's
       
    66     model/view architecture (including delegates).
       
    67 
       
    68     In this example, we will see how to implement a custom delegate
       
    69     to render and edit a "star rating" data type, which can store
       
    70     values such as "1 out of 5 stars".
       
    71 
       
    72     The example consists of the following classes:
       
    73 
       
    74     \list
       
    75     \o \c StarRating is the custom data type. It stores a rating
       
    76        expressed as stars, such as "2 out of 5 stars" or "5 out of
       
    77        6 stars".
       
    78 
       
    79     \o \c StarDelegate inherits QItemDelegate and provides support
       
    80        for \c StarRating (in addition to the data types already
       
    81        handled by QItemDelegate).
       
    82 
       
    83     \o \c StarEditor inherits QWidget and is used by \c StarDelegate
       
    84        to let the user edit a star rating using the mouse.
       
    85     \endlist
       
    86 
       
    87     To show the \c StarDelegate in action, we will fill a
       
    88     QTableWidget with some data and install the delegate on it.
       
    89 
       
    90     \section1 StarDelegate Class Definition
       
    91 
       
    92     Here's the definition of the \c StarDelegate class:
       
    93 
       
    94     \snippet examples/itemviews/stardelegate/stardelegate.h 0
       
    95 
       
    96     All public functions are reimplemented virtual functions from
       
    97     QItemDelegate to provide custom rendering and editing.
       
    98 
       
    99     \section1 StarDelegate Class Implementation
       
   100 
       
   101     The \l{QAbstractItemDelegate::}{paint()} function is
       
   102     reimplemented from QItemDelegate and is called whenever the view
       
   103     needs to repaint an item:
       
   104 
       
   105     \snippet examples/itemviews/stardelegate/stardelegate.cpp 0
       
   106 
       
   107     The function is invoked once for each item, represented by a
       
   108     QModelIndex object from the model. If the data stored in the item
       
   109     is a \c StarRating, we paint it ourselves; otherwise, we let
       
   110     QItemDelegate paint it for us. This ensures that the \c
       
   111     StarDelegate can handle the most common data types.
       
   112 
       
   113     In the case where the item is a \c StarRating, we draw the
       
   114     background if the item is selected, and we draw the item using \c
       
   115     StarRating::paint(), which we will review later.
       
   116 
       
   117     \c{StartRating}s can be stored in a QVariant thanks to the
       
   118     Q_DECLARE_METATYPE() macro appearing in \c starrating.h. More on
       
   119     this later.
       
   120 
       
   121     The \l{QAbstractItemDelegate::}{createEditor()} function is
       
   122     called when the user starts editing an item:
       
   123 
       
   124     \snippet examples/itemviews/stardelegate/stardelegate.cpp 2
       
   125 
       
   126     If the item is a \c StarRating, we create a \c StarEditor and
       
   127     connect its \c editingFinished() signal to our \c
       
   128     commitAndCloseEditor() slot, so we can update the model when the
       
   129     editor closes.
       
   130 
       
   131     Here's the implementation of \c commitAndCloseEditor():
       
   132 
       
   133     \snippet examples/itemviews/stardelegate/stardelegate.cpp 5
       
   134 
       
   135     When the user is done editing, we emit
       
   136     \l{QAbstractItemDelegate::}{commitData()} and
       
   137     \l{QAbstractItemDelegate::}{closeEditor()} (both declared in
       
   138     QAbstractItemDelegate), to tell the model that there is edited
       
   139     data and to inform the view that the editor is no longer needed.
       
   140 
       
   141     The \l{QAbstractItemDelegate::}{setEditorData()} function is
       
   142     called when an editor is created to initialize it with data
       
   143     from the model:
       
   144 
       
   145     \snippet examples/itemviews/stardelegate/stardelegate.cpp 3
       
   146 
       
   147     We simply call \c setStarRating() on the editor.
       
   148 
       
   149     The \l{QAbstractItemDelegate::}{setModelData()} function is
       
   150     called when editing is finished, to commit data from the editor
       
   151     to the model:
       
   152 
       
   153     \snippet examples/itemviews/stardelegate/stardelegate.cpp 4
       
   154 
       
   155     The \c sizeHint() function returns an item's preferred size:
       
   156 
       
   157     \snippet examples/itemviews/stardelegate/stardelegate.cpp 1
       
   158 
       
   159     We simply forward the call to \c StarRating.
       
   160 
       
   161     \section1 StarEditor Class Definition
       
   162 
       
   163     The \c StarEditor class was used when implementing \c
       
   164     StarDelegate. Here's the class definition:
       
   165 
       
   166     \snippet examples/itemviews/stardelegate/stareditor.h 0
       
   167 
       
   168     The class lets the user edit a \c StarRating by moving the mouse
       
   169     over the editor. It emits the \c editingFinished() signal when
       
   170     the user clicks on the editor.
       
   171 
       
   172     The protected functions are reimplemented from QWidget to handle
       
   173     mouse and paint events. The private function \c starAtPosition()
       
   174     is a helper function that returns the number of the star under
       
   175     the mouse pointer.
       
   176 
       
   177     \section1 StarEditor Class Implementation
       
   178 
       
   179     Let's start with the constructor:
       
   180 
       
   181     \snippet examples/itemviews/stardelegate/stareditor.cpp 0
       
   182 
       
   183     We enable \l{QWidget::setMouseTracking()}{mouse tracking} on the
       
   184     widget so we can follow the cursor even when the user doesn't
       
   185     hold down any mouse button. We also turn on QWidget's
       
   186     \l{QWidget::autoFillBackground}{auto-fill background} feature to
       
   187     obtain an opaque background. (Without the call, the view's
       
   188     background would shine through the editor.)
       
   189 
       
   190     The \l{QWidget::}{paintEvent()} function is reimplemented from
       
   191     QWidget:
       
   192 
       
   193     \snippet examples/itemviews/stardelegate/stareditor.cpp 1
       
   194 
       
   195     We simply call \c StarRating::paint() to draw the stars, just
       
   196     like we did when implementing \c StarDelegate.
       
   197 
       
   198     \snippet examples/itemviews/stardelegate/stareditor.cpp 2
       
   199 
       
   200     In the mouse event handler, we call \c setStarCount() on the
       
   201     private data member \c myStarRating to reflect the current cursor
       
   202     position, and we call QWidget::update() to force a repaint.
       
   203 
       
   204     \snippet examples/itemviews/stardelegate/stareditor.cpp 3
       
   205 
       
   206     When the user releases a mouse button, we simply emit the \c
       
   207     editingFinished() signal.
       
   208 
       
   209     \snippet examples/itemviews/stardelegate/stareditor.cpp 4
       
   210 
       
   211     The \c starAtPosition() function uses basic linear algebra to
       
   212     find out which star is under the cursor.
       
   213 
       
   214     \section1 StarRating Class Definition
       
   215 
       
   216     \snippet examples/itemviews/stardelegate/starrating.h 0
       
   217     \codeline
       
   218     \snippet examples/itemviews/stardelegate/starrating.h 1
       
   219 
       
   220     The \c StarRating class represents a rating as a number of stars.
       
   221     In addition to holding the data, it is also capable of painting
       
   222     the stars on a QPaintDevice, which in this example is either a
       
   223     view or an editor. The \c myStarCount member variable stores the
       
   224     current rating, and \c myMaxStarCount stores the highest possible
       
   225     rating (typically 5).
       
   226 
       
   227     The Q_DECLARE_METATYPE() macro makes the type \c StarRating known
       
   228     to QVariant, making it possible to store \c StarRating values in
       
   229     QVariant.
       
   230 
       
   231     \section1 StarRating Class Implementation
       
   232 
       
   233     The constructor initializes \c myStarCount and \c myMaxStarCount,
       
   234     and sets up the polygons used to draw stars and diamonds:
       
   235 
       
   236     \snippet examples/itemviews/stardelegate/starrating.cpp 0
       
   237 
       
   238     The \c paint() function paints the stars in this \c StarRating
       
   239     object on a paint device:
       
   240 
       
   241     \snippet examples/itemviews/stardelegate/starrating.cpp 2
       
   242 
       
   243     We first set the pen and brush we will use for painting. The \c
       
   244     mode parameter can be either \c Editable or \c ReadOnly. If \c
       
   245     mode is editable, we use the \l{QPalette::}{Highlight} color
       
   246     instead of the \l{QPalette::}{Foreground} color to draw the
       
   247     stars.
       
   248 
       
   249     Then we draw the stars. If we are in \c Edit mode, we paint
       
   250     diamonds in place of stars if the rating is less than the highest
       
   251     rating.
       
   252 
       
   253     The \c sizeHint() function returns the preferred size for an area
       
   254     to paint the stars on:
       
   255 
       
   256     \snippet examples/itemviews/stardelegate/starrating.cpp 1
       
   257 
       
   258     The preferred size is just enough to paint the maximum number of
       
   259     stars. The function is called by both \c StarDelegate::sizeHint()
       
   260     and \c StarEditor::sizeHint().
       
   261 
       
   262     \section1 The \c main() Function
       
   263 
       
   264     Here's the program's \c main() function:
       
   265 
       
   266     \snippet examples/itemviews/stardelegate/main.cpp 5
       
   267 
       
   268     The \c main() function creates a QTableWidget and sets a \c
       
   269     StarDelegate on it. \l{QAbstractItemView::}{DoubleClicked} and
       
   270     \l{QAbstractItemView::}{SelectedClicked} are set as
       
   271     \l{QAbstractItemView::editTriggers()}{edit triggers}, so that the
       
   272     editor is opened with a single click when the star rating item is
       
   273     selected.
       
   274     
       
   275     The \c populateTableWidget() function fills the QTableWidget with
       
   276     data:
       
   277 
       
   278     \snippet examples/itemviews/stardelegate/main.cpp 0
       
   279     \snippet examples/itemviews/stardelegate/main.cpp 1
       
   280     \dots
       
   281     \snippet examples/itemviews/stardelegate/main.cpp 2
       
   282     \snippet examples/itemviews/stardelegate/main.cpp 3
       
   283     \codeline
       
   284     \snippet examples/itemviews/stardelegate/main.cpp 4
       
   285 
       
   286     Notice the call to qVariantFromValue to convert a \c
       
   287     StarRating to a QVariant.
       
   288 
       
   289     \section1 Possible Extensions and Suggestions
       
   290 
       
   291     There are many ways to customize Qt's \l{Model/View
       
   292     Programming}{model/view framework}. The approach used in this
       
   293     example is appropriate for most custom delegates and editors.
       
   294     Examples of possibilities not used by the star delegate and star
       
   295     editor are:
       
   296 
       
   297     \list
       
   298     \o It is possible to open editors programmatically by calling
       
   299        QAbstractItemView::edit(), instead of relying on edit
       
   300        triggers. This could be use to support other edit triggers
       
   301        than those offered by the QAbstractItemView::EditTrigger enum.
       
   302        For example, in the Star Delegate example, hovering over an
       
   303        item with the mouse might make sense as a way to pop up an
       
   304        editor.
       
   305 
       
   306     \o By reimplementing QAbstractItemDelegate::editorEvent(), it is
       
   307        possible to implement the editor directly in the delegate,
       
   308        instead of creating a separate QWidget subclass.
       
   309     \endlist
       
   310 */