summaryrefslogtreecommitdiffstats
path: root/doc/src/examples/messageviewer.qdoc
blob: a8d7f6c270e16ce60c2227878a091cd44cf7a49d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
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
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
/****************************************************************************
**
** Copyright (C) 2015 The Qt Company Ltd.
** Contact: http://www.qt-project.org/legal
**
** This file is part of the documentation of the Qt Toolkit.
**
** $QT_BEGIN_LICENSE:FDL$
** Commercial License Usage
** Licensees holding valid commercial Qt licenses may use this file in
** accordance with the commercial license agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and The Qt Company. For licensing terms
** and conditions see https://www.qt.io/terms-conditions. For further
** information use the contact form at https://www.qt.io/contact-us.
**
** GNU Free Documentation License Usage
** Alternatively, this file may be used under the terms of the GNU Free
** Documentation License version 1.3 as published by the Free Software
** Foundation and appearing in the file included in the packaging of
** this file. Please review the following information to ensure
** the GNU Free Documentation License version 1.3 requirements
** will be met: https://www.gnu.org/licenses/fdl-1.3.html.
** $QT_END_LICENSE$
**
****************************************************************************/

/*!
    \example messageviewer
    \title MessageViewer Example

    The MessageViewer example shows how to create a client application
    which accesses messages stored by the Messaging Framework. The
    example application searches for, and displays the properties of messages 
    stored by the Messaging Framework, but uses Qt Extended services to delegate the display 
    of the messages to another application.

    \image messageviewer-example.png Screenshot of the MessageViewer example

    The example application uses the Qt Extended PIM library's Contacts listing 
    to list the messages stored on the device, by contact.  The application 
    starts by allowing the user to select a contact from those stored on 
    the device.  Once a contact has been selected, a list of all the 
    messages exchanged with that contact is displayed.  When a message is 
    selected from this list, the application creates a QtopiaServiceRequest
    to request that another application display the content of the
    selected message.

    The application is structured as a stack of widgets, contained by a QStackedWidget.
    We need two widgets in our stack: one to select a contact from a list, and
    another to select a message from a list.  

    The contact selector widget uses a QContactListView to present a list
    of available contacts:

    \quotefromfile messageviewer/messageviewer.cpp
    \skipto class ContactSelector
    \printto Adjust ContactSelector
    \skipuntil end-Adjust
    \printuntil };

    Once we have selected a contact, we will display a listing of the messages
    that have been exchanged with the contact.  For this, we need a stackable
    widget to list messages:

    \quotefromfile messageviewer/messageviewer.cpp
    \skipto class MessageSelector
    \printto Adjust MessageSelector
    \skipuntil end-Adjust
    \printuntil };

    The MessageSelector uses a QListView to present a listing of messages
    associated with a specific contact. To do this, we need to create a 
    model of message data that the QListView will present. Our class 
    therefore contains a model object, of type MessageModel, and an object
    of type MessageDelegate, whose responsibility is to render each element
    in the list view using the data of each message.

    The MessageModel class is derived from QStandardItemModel, which is
    extended with a simple interface.  With setContact(), we supply the 
    model with a QContact, and it creates a listing of messages 
    associated with that contact. It also provides the utility function 
    messageId() which we use to extract the identifier of a message from the
    listing, given an index into the model.

    \quotefromfile messageviewer/messagemodel.h
    \skipto class MessageModel
    \printuntil };

    The Messaging Framework Client library uses QMailMessageId objects to identify messages
    stored in the system.  QMailMessageId objects can be default-constructed to 
    an uninitialised state that does not identify any message, or they can
    contain the identifier for any message stored in the device. When we
    wish to view a message, the identifier for that message is all that 
    we will need in order to request that another application display
    the message.

    The MessageModel class uses a helper class to contain the data elements
    relevant to each individual message that we will list.  The helper class,
    MessageItem, is derived from QStandardItem:

    \quotefromfile messageviewer/messagemodel.cpp
    \skipto class MessageItem
    \printuntil };

    The MessageModel::setContact() member function does the work of finding 
    messages associated with a contact, and adding them to our data model. To
    find messages, we use the QMailStore::queryMessages() function. One of
    the overloads of this function takes a QMailMessageKey parameter, which 
    contains the filtering information needed to locate a subset of the 
    device's messages. A QMailMessageKey contains three data elements: a 
    message property, a value, and a relation to compare the value to the 
    specified property of each message. More complicated filters can be 
    created by logically composing QMailMessageKey objects, with AND and 
    OR operators.

    For our application, we want to find messages that are related to the
    contact that the user has chosen.  So, we will find messages that fit into
    either of two categories: those that were sent to the contact, and those that 
    were received from the contact.

    \skipto MessageModel::setContact
    \printuntil msgsTo;

    The contact may have numerous phone numbers, so we pass over the list of
    phone numbers stored for this contact, and for each phone number, we add 
    another filter criterion to our query.  For messages we received from the 
    contact, we add a QMailMessageKey matching where the message's \c Sender 
    property is equal to the current phone number (note that the \c Equal 
    relation is the default, so we don't need to specify it explicitly.) 
    For messages that we sent to the contact, we add a QMailMessageKey matching 
    where the message's \c Recipients property contains the current number.  
    Unlike \c Sender, a messages's \c Recipients property can contain multiple 
    contact's addresses, so we need to use the \c Includes relation instead of 
    the default \c Equal.

    \skipto s phone numbers
    \printuntil }

    We also want to locate messages exchanged with this contact using email,
    so we add further criteria to our filters for each email address that 
    is stored for this contact:

    \skipto s email addresses
    \printuntil }

    Now we have created the message filters we want, we use the 
    QMailStore::queryMessages() function to locate the matching messages. This
    function applies the criteria of the supplied QMailMessageKey to each 
    message stored by the Messaging Framework, and returns a list 
    containing the QMailMessageId of each matching message. An overload of the
    function takes a second parameter: a QMailMessageSortKey which 
    determines the order in which the matching messages are returned. We
    will use this option to sort the messages into reverse chronological 
    order.

    \printto Add each

    We then take each QMailMessageId from the list, and create a new MessageItem
    object from each one, to add to our model:

    \printuntil }

    The MessageItem class stores the data we need to present each message 
    in our QListView. In order to present a useful listing of the message, 
    we need to extract some data from the message.  We do this using the
    QMailMessageMetaData class, which has a constructor taking a QMailMessageId.
    Since we only need summary information to present the message in the 
    list view, we only want to load the meta data for the message identified 
    by the QMailMessageId.  If we needed to access the content of the 
    message, we would instead need to create an instance of the QMailMessage class.

    Once we have loaded the message information, we extract some useful 
    elements (such as the message subject, the message timestamp, and the 
    type of the message), and store those items for the QListView delegate 
    to use when rendering the message item:

    \quotefromfile messageviewer/messagemodel.cpp
    \skipto MessageItem::MessageItem
    \printuntil }

    The MessageDelegate class simply takes the data items stored by each 
    MessageItem instance, and uses them to render a representation of
    the message.

    Now that we have widgets to list contacts and messages, we simply need 
    to connect them together:
    
    \quotefromfile messageviewer/messageviewer.cpp
    \skipto MessageViewer::MessageViewer
    \printto Adjust MessageViewer
    \skipuntil end-Adjust
    \printuntil }

    \skipto MessageViewer::showContactList
    \printuntil }

    When we have a contact selected from our contact list, we create a 
    list of messages for that contact. This is handled 
    by the MessageSelector::listMessages() function:

    \quotefromfile messageviewer/messageviewer.cpp
    \skipto void MessageSelector::listMessages(
    \printto end-listMessages

    When the message list has been prepared, we move our message list to 
    the top of the widget stack, in MessageViewer::showMessageList():

    \skipto void MessageViewer::showMessageList(
    \printuntil }

    Finally, we handle the event where the user selects a message from our
    list. Rather than displaying the message ourself, we will use the 
    QtopiaServiceRequest mechanism to request that another application 
    handle this task for us. The \c Messages service exports a \c viewMessage
    function, which takes a QMailMessageId object as a parameter; we respond to the
    user's selection by invoking this service with the identifier of the
    message that they selected:

    \skipto void MessageViewer::viewMessage(
    \printuntil }

    \sa {MessageNavigator Example}
 */