QGCFileDialog.cc 7.64 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
/*=====================================================================
 
 QGroundControl Open Source Ground Control Station
 
 (c) 2009 - 2014 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/>.
 
 ======================================================================*/

#include "QGCFileDialog.h"
#include "QGCApplication.h"
26
#include "MainWindow.h"
27

28
#ifdef QT_DEBUG
dogmaphobic's avatar
dogmaphobic committed
29
#ifndef __mobile__
30 31
#include "UnitTest.h"
#endif
dogmaphobic's avatar
dogmaphobic committed
32
#endif
33

Don Gagne's avatar
Don Gagne committed
34 35
#include <QRegularExpression>
#include <QMessageBox>
36
#include <QPushButton>
Don Gagne's avatar
Don Gagne committed
37

38 39 40 41 42
QString QGCFileDialog::getExistingDirectory(
    QWidget* parent,
    const QString& caption,
    const QString& dir,
    Options options)
43
{
44
    _validate(options);
45 46
    
#ifdef QT_DEBUG
dogmaphobic's avatar
dogmaphobic committed
47
#ifndef __mobile__
48 49 50
    if (qgcApp()->runningUnitTests()) {
        return UnitTest::_getExistingDirectory(parent, caption, dir, options);
    } else
dogmaphobic's avatar
dogmaphobic committed
51
#endif
52 53 54 55 56 57
#endif
    {
        return QFileDialog::getExistingDirectory(parent, caption, dir, options);
    }
}

58 59 60 61 62 63
QString QGCFileDialog::getOpenFileName(
    QWidget* parent,
    const QString& caption,
    const QString& dir,
    const QString& filter,
    Options options)
64
{
65
    _validate(options);
66 67
    
#ifdef QT_DEBUG
dogmaphobic's avatar
dogmaphobic committed
68
#ifndef __mobile__
69
    if (qgcApp()->runningUnitTests()) {
70
        return UnitTest::_getOpenFileName(parent, caption, dir, filter, options);
71
    } else
dogmaphobic's avatar
dogmaphobic committed
72
#endif
73 74
#endif
    {
75
        return QFileDialog::getOpenFileName(parent, caption, dir, filter, NULL, options);
76 77 78
    }
}

79 80 81 82 83 84
QStringList QGCFileDialog::getOpenFileNames(
    QWidget* parent,
    const QString& caption,
    const QString& dir,
    const QString& filter,
    Options options)
85
{
86
    _validate(options);
87 88
    
#ifdef QT_DEBUG
dogmaphobic's avatar
dogmaphobic committed
89
#ifndef __mobile__
90
    if (qgcApp()->runningUnitTests()) {
91
        return UnitTest::_getOpenFileNames(parent, caption, dir, filter, options);
92
    } else
dogmaphobic's avatar
dogmaphobic committed
93
#endif
94 95
#endif
    {
96
        return QFileDialog::getOpenFileNames(parent, caption, dir, filter, NULL, options);
97 98 99
    }
}

100 101
QString QGCFileDialog::getSaveFileName(
    QWidget* parent,
102 103 104
    const QString& caption,
    const QString& dir,
    const QString& filter,
105
    const QString& defaultSuffix,
106
    bool strict,
107
    Options options)
108
{
109
    _validate(options);
110

111
#ifdef QT_DEBUG
dogmaphobic's avatar
dogmaphobic committed
112
#ifndef __mobile__
113
    if (qgcApp()->runningUnitTests()) {
114
        return UnitTest::_getSaveFileName(parent, caption, dir, filter, defaultSuffix, options);
115
    } else
dogmaphobic's avatar
dogmaphobic committed
116
#endif
117 118
#endif
    {
119
        QString defaultSuffixCopy(defaultSuffix);
120
        QFileDialog dlg(parent, caption, dir, filter);
121
        dlg.setAcceptMode(QFileDialog::AcceptSave);
122
        if (options) {
123
            dlg.setOptions(options);
124
        }
125
        if (!defaultSuffixCopy.isEmpty()) {
126
            //-- Make sure dot is not present
127 128
            if (defaultSuffixCopy.startsWith(".")) {
                defaultSuffixCopy.remove(0,1);
129
            }
130
            dlg.setDefaultSuffix(defaultSuffixCopy);
131
        }
132 133 134
        while (true) {
            if (dlg.exec()) {
                if (dlg.selectedFiles().count()) {
135 136 137 138 139 140 141 142 143 144 145 146
                    QString result = dlg.selectedFiles().first();
                    //-- If we don't care about the extension, just return it
                    if (!strict) {
                        return result;
                    } else {
                        //-- We must enforce the file extension
                        QFileInfo fi(result);
                        QString userSuffix(fi.suffix());
                        if (_validateExtension(filter, userSuffix)) {
                            return result;
                        }
                        //-- Do we have a default extension?
147
                        if (defaultSuffixCopy.isEmpty()) {
148
                            //-- We don't, so get the first one in the filter
149
                            defaultSuffixCopy = _getFirstExtensionInFilter(filter);
150
                        }
151 152
                        //-- If this is set to strict, we have to have a default extension
                        Q_ASSERT(defaultSuffixCopy.isEmpty() == false);
153 154
                        //-- Forcefully append our desired extension
                        result += ".";
155
                        result += defaultSuffixCopy;
156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176
                        //-- Check and see if this new file already exists
                        fi.setFile(result);
                        if (fi.exists()) {
                            //-- Ask user what to do
                            QMessageBox msgBox(
                                QMessageBox::Warning,
                                tr("File Exists"),
                                tr("%1 already exists.\nDo you want to replace it?").arg(fi.fileName()),
                                QMessageBox::Cancel,
                                parent);
                            msgBox.addButton(QMessageBox::Retry);
                            QPushButton *overwriteButton = msgBox.addButton(tr("Replace"), QMessageBox::ActionRole);
                            msgBox.setDefaultButton(QMessageBox::Retry);
                            msgBox.setWindowModality(Qt::ApplicationModal);
                            if (msgBox.exec() == QMessageBox::Retry) {
                                continue;
                            } else if (msgBox.clickedButton() == overwriteButton) {
                                return result;
                            }
                        } else {
                            return result;
177 178
                        }
                    }
179
                }
180
            }
181
            break;
182
        }
183
        return QString("");
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
/// @brief Make sure filename is using one of the valid extensions defined in the filter
bool QGCFileDialog::_validateExtension(const QString& filter, const QString& extension) {
    QRegularExpression re("(\\*\\.\\w+)");
    QRegularExpressionMatchIterator i = re.globalMatch(filter);
    while (i.hasNext()) {
        QRegularExpressionMatch match = i.next();
        if (match.hasMatch()) {
            //-- Compare "foo" with "*.foo"
            if(extension == match.captured(0).mid(2)) {
                return true;
            }
        }
    }
    return false;
}

/// @brief Returns first extension found in filter
QString QGCFileDialog::_getFirstExtensionInFilter(const QString& filter) {
    QRegularExpression re("(\\*\\.\\w+)");
    QRegularExpressionMatchIterator i = re.globalMatch(filter);
    while (i.hasNext()) {
        QRegularExpressionMatch match = i.next();
        if (match.hasMatch()) {
210 211
            //-- Return "foo" from "*.foo"
            return match.captured(0).mid(2);
212 213 214 215 216
        }
    }
    return QString("");
}

217
/// @brief Validates and updates the parameters for the file dialog calls
218
void QGCFileDialog::_validate(Options& options)
219 220 221 222
{
    // You can't use QGCFileDialog if QGCApplication is not created yet.
    Q_ASSERT(qgcApp());
    
223
    Q_ASSERT_X(QThread::currentThread() == qgcApp()->thread(), "Threading issue", "QGCFileDialog can only be called from main thread");
dogmaphobic's avatar
dogmaphobic committed
224
#ifdef __mobile__
dogmaphobic's avatar
dogmaphobic committed
225 226
    Q_UNUSED(options)
#else
227
    // On OSX native dialog can hang so we always use Qt dialogs
Don Gagne's avatar
Don Gagne committed
228
    options |= DontUseNativeDialog;
dogmaphobic's avatar
dogmaphobic committed
229
#endif
230 231
    if (MainWindow::instance()) {
    }
232
}