source: trunk/examples/webkit/imageanalyzer/imageanalyzer.cpp@ 1001

Last change on this file since 1001 was 846, checked in by Dmitry A. Kuminov, 14 years ago

trunk: Merged in qt 4.7.2 sources from branches/vendor/nokia/qt.

File size: 7.5 KB
Line 
1/****************************************************************************
2**
3** Copyright (C) 2011 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 examples of the Qt Toolkit.
8**
9** $QT_BEGIN_LICENSE:BSD$
10** You may use this file under the terms of the BSD license as follows:
11**
12** "Redistribution and use in source and binary forms, with or without
13** modification, are permitted provided that the following conditions are
14** met:
15** * Redistributions of source code must retain the above copyright
16** notice, this list of conditions and the following disclaimer.
17** * Redistributions in binary form must reproduce the above copyright
18** notice, this list of conditions and the following disclaimer in
19** the documentation and/or other materials provided with the
20** distribution.
21** * Neither the name of Nokia Corporation and its Subsidiary(-ies) nor
22** the names of its contributors may be used to endorse or promote
23** products derived from this software without specific prior written
24** permission.
25**
26** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
27** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
28** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
29** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
30** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
31** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
32** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
33** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
34** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
35** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
36** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."
37** $QT_END_LICENSE$
38**
39****************************************************************************/
40
41
42#include <QNetworkReply>
43#include <QNetworkRequest>
44#include <QNetworkAccessManager>
45#include <QNetworkDiskCache>
46#include "imageanalyzer.h"
47
48/*!
49 * This class operates as follows:
50 * Parent calls the slot startAnalysis which shoves a list of QStrings into URLQueue and then calls fetchURLs.
51 * FetchURLs sends out HTTP GETs for each image it can't get out of the cache.
52 * As the responses come in, handleReply tries to create an image out of each and pushes those images into imageQueue.
53 * On the last (detected by no outstandingFetches and URLQueue.isEmpty()) call to queueImage (from handleReply)
54 * a thread is forked to process all the images. When it finishes, it emits a finished signal that is received
55 * by our JavaScript code.
56 */
57
58//! [ ImageAnalyzer - Constructor ]
59ImageAnalyzer::ImageAnalyzer(QNetworkDiskCache* netcache, QObject* parent)
60 : QObject(parent), m_cache(netcache), m_outstandingFetches(0)
61{
62 /* ImageAnalyzer only wants to receive http responses
63 for requests that it makes, so that's why it has its own
64 QNetworkAccessManager. */
65 m_network = new QNetworkAccessManager(this);
66 m_watcher = new QFutureWatcher<QRgb>(this);
67 /* We want to share a cache with the web browser,
68 in case it has some images we want: */
69 m_network->setCache(m_cache);
70
71 QObject::connect(m_network, SIGNAL(finished(QNetworkReply*)),
72 this, SLOT(handleReply(QNetworkReply*)));
73 QObject::connect(m_watcher, SIGNAL(finished()),
74 this, SLOT(doneProcessing()));
75 QObject::connect(m_watcher, SIGNAL(progressValueChanged(int)),
76 this, SLOT(progressStatus(int)));
77}
78//! [ ImageAnalyzer - Constructor ]
79ImageAnalyzer::~ImageAnalyzer()
80{
81 delete(m_watcher);
82}
83
84
85QRgb ImageAnalyzer::lastResults()
86{
87 int rTot = 0;
88 int bTot = 0;
89 int gTot = 0;
90 int count = m_watcher->future().results().size();
91 foreach(const QRgb & triplet, m_watcher->future().results())
92 {
93 rTot += qRed(triplet);
94 bTot += qBlue(triplet);
95 gTot += qGreen(triplet);
96 }
97 return qRgb(rTot/count, bTot/count, gTot/count);
98}
99
100float ImageAnalyzer::lastRed() { return qRed(lastResults())/2.55; }
101float ImageAnalyzer::lastGreen() { return qGreen(lastResults())/2.55; }
102float ImageAnalyzer::lastBlue() { return qBlue(lastResults())/2.55; }
103
104void ImageAnalyzer::progressStatus(int newstat)
105{
106 emit updateProgress(newstat, m_watcher->progressMaximum());
107}
108
109
110bool ImageAnalyzer::isBusy()
111{
112 return m_watcher->isRunning();
113}
114
115
116//! [ ImageAnalyzer - startAnalysis ]
117void ImageAnalyzer::startAnalysis(const QStringList & urls)
118{
119 m_URLQueue = urls;
120 fetchURLs();
121}
122//! [ ImageAnalyzer - startAnalysis ]
123
124/*!
125 * Analyzes the entire queue - just starts all our http GETs.
126 */
127//! [ ImageAnalyzer - fetchURLs ]
128void ImageAnalyzer::fetchURLs()
129{
130 while (!m_URLQueue.isEmpty())
131 {
132 QString url = m_URLQueue.takeFirst();
133 QUrl URL = QUrl(url);
134 QIODevice * pData = m_cache->data(URL);
135 // Is image already loaded in cache?
136 if (pData == 0) {
137 // HTTP Get image over network.
138 m_outstandingFetches++;
139 QNetworkRequest request = QNetworkRequest(URL);
140 request.setRawHeader("User-Agent", "Nokia - Custom QT app");
141 m_network->get(request);
142 } else {
143 // Get image from cache
144 QImage image;
145 image.load(pData, 0);
146 if (!image.isNull())
147 queueImage(image);
148 delete(pData);
149 }
150 }
151}
152//! [ ImageAnalyzer - fetchURLs ]
153/*
154 * Slot to handle the incoming responses from our http GETs
155 */
156//! [ ImageAnalyzer - handleReply ]
157void ImageAnalyzer::handleReply(QNetworkReply * pReply)
158{
159 m_outstandingFetches--;
160 if (pReply->error()) {
161 qDebug() << "Error code" << pReply->error();
162 qDebug() << "Http code" << pReply->attribute(QNetworkRequest::HttpStatusCodeAttribute);
163 return;
164 }
165 QImage image;
166 image.load(pReply, 0);
167 pReply->deleteLater();
168 if (image.isNull()) {
169 qDebug() << "bad image";
170 qDebug() << pReply->rawHeaderList();
171 foreach(QByteArray element, pReply->rawHeaderList()) {
172 qDebug() << element << " = " << pReply->rawHeader(element);
173 }
174 return;
175 }
176 queueImage(image);
177}
178//! [ ImageAnalyzer - handleReply ]
179
180void ImageAnalyzer::doneProcessing()
181{
182 m_imageQueue = QList<QImage>();
183 emit finishedAnalysis();
184}
185//! [ ImageAnalyzer - queueImage ]
186void ImageAnalyzer::queueImage(QImage img)
187{
188 if (!img.isNull())
189 m_imageQueue << img;
190
191 if (m_outstandingFetches == 0 && m_URLQueue.isEmpty()) {
192 m_watcher->setFuture(QtConcurrent::mapped(m_imageQueue, averageRGB));
193 }
194}
195//! [ ImageAnalyzer - queueImage ]
196
197//! [ ImageAnalyzer - averageRGB ]
198QRgb averageRGB(const QImage &img)
199{
200 int pixelCount = img.width() * img.height();
201 int rAvg, gAvg, bAvg;
202
203 // We waste some time here:
204 for (int timeWaster=0; timeWaster < 100; timeWaster++) {
205 quint64 rTot = 0;
206 quint64 gTot = 0;
207 quint64 bTot = 0;
208 for (int i=0; i < img.width(); i++) {
209 for (int j=0; j < img.height(); j++) {
210 QRgb pixel = img.pixel(i,j);
211 rTot += qRed(pixel);
212 gTot += qGreen(pixel);
213 bTot += qBlue(pixel);
214 }
215 }
216 rAvg = (rTot)/(pixelCount);
217 gAvg = (gTot)/(pixelCount);
218 bAvg = (bTot)/(pixelCount);
219 }
220 return qRgb(rAvg, gAvg, bAvg);
221}
222//! [ ImageAnalyzer - averageRGB ]
Note: See TracBrowser for help on using the repository browser.