WebImageCache.cc 5.24 KB
Newer Older
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
/*=====================================================================

QGroundControl Open Source Ground Control Station

(c) 2009, 2010 QGROUNDCONTROL PROJECT <http://www.qgroundcontrol.org>

This file is part of the QGROUNDCONTROL project

    QGROUNDCONTROL is free software: you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
    the Free Software Foundation, either version 3 of the License, or
    (at your option) any later version.

    QGROUNDCONTROL is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with QGROUNDCONTROL. If not, see <http://www.gnu.org/licenses/>.

======================================================================*/

/**
 * @file
 *   @brief Definition of the class WebImageCache.
 *
 *   @author Lionel Heng <hengli@student.ethz.ch>
 *
 */

32 33 34 35 36 37 38 39 40 41 42
#include "WebImageCache.h"

#include <QNetworkReply>
#include <QPixmap>

WebImageCache::WebImageCache(QObject* parent, uint32_t _cacheSize)
    : QObject(parent)
    , cacheSize(_cacheSize)
    , currentReference(0)
    , networkManager(new QNetworkAccessManager)
{
43 44 45 46 47 48
    for (uint32_t i = 0; i < cacheSize; ++i)
    {
        WebImagePtr image(new WebImage);

        webImages.push_back(image);
    }
49 50 51 52 53 54

    connect(networkManager.data(), SIGNAL(finished(QNetworkReply*)),
            this, SLOT(downloadFinished(QNetworkReply*)));
}

QPair<WebImagePtr, int32_t>
55
WebImageCache::lookup(const QString& url, bool useHeightModel)
56
{
57 58
    QPair<WebImagePtr, int32_t> cacheEntry;

59 60 61
    for (int32_t i = 0; i < webImages.size(); ++i)
    {
        if (webImages[i]->getState() != WebImage::UNINITIALIZED &&
62 63
            webImages[i]->getSourceURL() == url &&
            webImages[i]->is3D() == useHeightModel)
64
        {
65 66
            cacheEntry.first = webImages[i];
            cacheEntry.second = i;
67 68 69 70
            break;
        }
    }

71
    if (cacheEntry.first.isNull())
72 73 74 75 76 77
    {
        for (int32_t i = 0; i < webImages.size(); ++i)
        {
            // get uninitialized image
            if (webImages[i]->getState() == WebImage::UNINITIALIZED)
            {
78 79
                cacheEntry.first = webImages[i];
                cacheEntry.second = i;
80 81 82 83
                break;
            }
            // get oldest image
            else if (webImages[i]->getState() == WebImage::READY &&
84 85 86
                     (cacheEntry.first.isNull() ||
                      webImages[i]->getLastReference() <
                      cacheEntry.first->getLastReference()))
87
            {
88 89
                cacheEntry.first = webImages[i];
                cacheEntry.second = i;
90 91 92
            }
        }

93
        if (cacheEntry.first.isNull())
94 95 96 97 98
        {
            return qMakePair(WebImagePtr(), -1);
        }
        else
        {
99
            if (cacheEntry.first->getState() == WebImage::READY)
100
            {
101
                cacheEntry.first->clear();
102
            }
103 104
            cacheEntry.first->setSourceURL(url);
            cacheEntry.first->setLastReference(currentReference);
105
            ++currentReference;
106
            cacheEntry.first->setState(WebImage::REQUESTED);
107

108 109 110 111 112 113
            if (url.left(4).compare("http") == 0)
            {
                networkManager->get(QNetworkRequest(QUrl(url)));
            }
            else
            {
114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129
                bool success;

                if (useHeightModel)
                {
                    QString heightURL = url;
                    heightURL.replace("color", "dom");
                    heightURL.replace(".jpg", ".txt");

                    success = cacheEntry.first->setData(url, heightURL);
                }
                else
                {
                    success = cacheEntry.first->setData(url);
                }

                if (success)
130 131 132 133 134 135 136 137 138
                {
                    cacheEntry.first->setSyncFlag(true);
                    cacheEntry.first->setState(WebImage::READY);
                }
                else
                {
                    cacheEntry.first->setState(WebImage::UNINITIALIZED);
                }
            }
139

140
            return cacheEntry;
141 142 143 144
        }
    }
    else
    {
145
        if (cacheEntry.first->getState() == WebImage::READY)
146
        {
147
            cacheEntry.first->setLastReference(currentReference);
148
            ++currentReference;
149
            return cacheEntry;
150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166
        }
        else
        {
            return qMakePair(WebImagePtr(), -1);
        }
    }
}

WebImagePtr
WebImageCache::at(int32_t index) const
{
    return webImages[index];
}

void
WebImageCache::downloadFinished(QNetworkReply* reply)
{
167 168 169 170
    reply->deleteLater();

    if (reply->error() != QNetworkReply::NoError)
    {
171 172 173 174 175 176 177 178
        return;
    }
    QVariant attribute = reply->attribute(QNetworkRequest::RedirectionTargetAttribute);
    if (attribute.isValid())
    {
        return;
    }

179 180 181 182 183 184 185 186
    WebImagePtr image;
    foreach(image, webImages)
    {
        if (reply->url().toString() == image->getSourceURL())
        {
            image->setData(reply->readAll());
            image->setSyncFlag(true);
            image->setState(WebImage::READY);
187

188 189 190
            return;
        }
    }
191
}