forked from qt-creator/qt-creator
672 lines
19 KiB
C++
672 lines
19 KiB
C++
|
|
/***************************************************************************
|
||
|
|
**
|
||
|
|
** This file is part of Qt Creator
|
||
|
|
**
|
||
|
|
** Copyright (c) 2008 Nokia Corporation and/or its subsidiary(-ies).
|
||
|
|
**
|
||
|
|
** Contact: Qt Software Information (qt-info@nokia.com)
|
||
|
|
**
|
||
|
|
**
|
||
|
|
** Non-Open Source Usage
|
||
|
|
**
|
||
|
|
** Licensees may use this file in accordance with the Qt Beta Version
|
||
|
|
** License Agreement, Agreement version 2.2 provided with the Software or,
|
||
|
|
** alternatively, in accordance with the terms contained in a written
|
||
|
|
** agreement between you and Nokia.
|
||
|
|
**
|
||
|
|
** GNU General Public License Usage
|
||
|
|
**
|
||
|
|
** Alternatively, this file may be used under the terms of the GNU General
|
||
|
|
** Public License versions 2.0 or 3.0 as published by the Free Software
|
||
|
|
** Foundation and appearing in the file LICENSE.GPL included in the packaging
|
||
|
|
** of this file. Please review the following information to ensure GNU
|
||
|
|
** General Public Licensing requirements will be met:
|
||
|
|
**
|
||
|
|
** http://www.fsf.org/licensing/licenses/info/GPLv2.html and
|
||
|
|
** http://www.gnu.org/copyleft/gpl.html.
|
||
|
|
**
|
||
|
|
** In addition, as a special exception, Nokia gives you certain additional
|
||
|
|
** rights. These rights are described in the Nokia Qt GPL Exception version
|
||
|
|
** 1.2, included in the file GPL_EXCEPTION.txt in this package.
|
||
|
|
**
|
||
|
|
***************************************************************************/
|
||
|
|
#include "basefilewizard.h"
|
||
|
|
#include "mimedatabase.h"
|
||
|
|
|
||
|
|
#include <coreplugin/editormanager/editormanager.h>
|
||
|
|
#include <coreplugin/icore.h>
|
||
|
|
#include <coreplugin/coreconstants.h>
|
||
|
|
#include <coreplugin/ifilewizardextension.h>
|
||
|
|
#include <utils/filewizarddialog.h>
|
||
|
|
|
||
|
|
#include <QtCore/QDir>
|
||
|
|
#include <QtCore/QFile>
|
||
|
|
#include <QtCore/QFileInfo>
|
||
|
|
#include <QtCore/QVector>
|
||
|
|
#include <QtCore/QDebug>
|
||
|
|
#include <QtCore/QSharedData>
|
||
|
|
#include <QtCore/QEventLoop>
|
||
|
|
#include <QtCore/QSharedPointer>
|
||
|
|
|
||
|
|
#include <QtGui/QMessageBox>
|
||
|
|
#include <QtGui/QWizard>
|
||
|
|
#include <QtGui/QMainWindow>
|
||
|
|
|
||
|
|
enum { debugWizard = 0 };
|
||
|
|
|
||
|
|
namespace Core {
|
||
|
|
|
||
|
|
class GeneratedFilePrivate : public QSharedData {
|
||
|
|
public:
|
||
|
|
GeneratedFilePrivate() {}
|
||
|
|
explicit GeneratedFilePrivate(const QString &p);
|
||
|
|
QString path;
|
||
|
|
QString contents;
|
||
|
|
QString editorKind;
|
||
|
|
};
|
||
|
|
|
||
|
|
GeneratedFilePrivate::GeneratedFilePrivate(const QString &p) :
|
||
|
|
path(p)
|
||
|
|
{
|
||
|
|
}
|
||
|
|
|
||
|
|
GeneratedFile::GeneratedFile() :
|
||
|
|
m_d(new GeneratedFilePrivate)
|
||
|
|
{
|
||
|
|
}
|
||
|
|
|
||
|
|
GeneratedFile::GeneratedFile(const QString &p) :
|
||
|
|
m_d(new GeneratedFilePrivate(p))
|
||
|
|
{
|
||
|
|
}
|
||
|
|
|
||
|
|
GeneratedFile::GeneratedFile(const GeneratedFile &rhs) :
|
||
|
|
m_d(rhs.m_d)
|
||
|
|
{
|
||
|
|
}
|
||
|
|
|
||
|
|
GeneratedFile &GeneratedFile::operator=(const GeneratedFile &rhs)
|
||
|
|
{
|
||
|
|
if (this != &rhs)
|
||
|
|
m_d.operator=(rhs.m_d);
|
||
|
|
return *this;
|
||
|
|
}
|
||
|
|
|
||
|
|
GeneratedFile::~GeneratedFile()
|
||
|
|
{
|
||
|
|
}
|
||
|
|
|
||
|
|
QString GeneratedFile::path() const
|
||
|
|
{
|
||
|
|
return m_d->path;
|
||
|
|
}
|
||
|
|
|
||
|
|
void GeneratedFile::setPath(const QString &p)
|
||
|
|
{
|
||
|
|
m_d->path = p;
|
||
|
|
}
|
||
|
|
|
||
|
|
QString GeneratedFile::contents() const
|
||
|
|
{
|
||
|
|
return m_d->contents;
|
||
|
|
}
|
||
|
|
|
||
|
|
void GeneratedFile::setContents(const QString &c)
|
||
|
|
{
|
||
|
|
m_d->contents = c;
|
||
|
|
}
|
||
|
|
|
||
|
|
QString GeneratedFile::editorKind() const
|
||
|
|
{
|
||
|
|
return m_d->editorKind;
|
||
|
|
}
|
||
|
|
|
||
|
|
void GeneratedFile::setEditorKind(const QString &k)
|
||
|
|
{
|
||
|
|
m_d->editorKind = k;
|
||
|
|
}
|
||
|
|
|
||
|
|
bool GeneratedFile::write(QString *errorMessage) const
|
||
|
|
{
|
||
|
|
// Ensure the directory
|
||
|
|
const QFileInfo info(m_d->path);
|
||
|
|
const QDir dir = info.absoluteDir();
|
||
|
|
if (!dir.exists()) {
|
||
|
|
if (!dir.mkpath(dir.absolutePath())) {
|
||
|
|
*errorMessage = BaseFileWizard::tr("Unable to create the directory %1.").arg(dir.absolutePath());
|
||
|
|
return false;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
// Write out
|
||
|
|
QFile file(m_d->path);
|
||
|
|
if (!file.open(QIODevice::WriteOnly|QIODevice::Text)) {
|
||
|
|
*errorMessage = BaseFileWizard::tr("Unable to open %1 for writing: %2").arg(m_d->path, file.errorString());
|
||
|
|
return false;
|
||
|
|
}
|
||
|
|
if (file.write(m_d->contents.toUtf8()) == -1) {
|
||
|
|
*errorMessage = BaseFileWizard::tr("Error while writing to %1: %2").arg(m_d->path, file.errorString());
|
||
|
|
return false;
|
||
|
|
}
|
||
|
|
file.close();
|
||
|
|
return true;
|
||
|
|
}
|
||
|
|
// ------------ BaseFileWizardParameterData
|
||
|
|
class BaseFileWizardParameterData : public QSharedData {
|
||
|
|
public:
|
||
|
|
explicit BaseFileWizardParameterData(IWizard::Kind kind = IWizard::FileWizard);
|
||
|
|
|
||
|
|
IWizard::Kind kind;
|
||
|
|
QIcon icon;
|
||
|
|
QString description;
|
||
|
|
QString name;
|
||
|
|
QString category;
|
||
|
|
QString trCategory;
|
||
|
|
};
|
||
|
|
|
||
|
|
BaseFileWizardParameterData::BaseFileWizardParameterData(IWizard::Kind k) :
|
||
|
|
kind(k)
|
||
|
|
{
|
||
|
|
}
|
||
|
|
|
||
|
|
BaseFileWizardParameters::BaseFileWizardParameters(IWizard::Kind kind) :
|
||
|
|
m_d(new BaseFileWizardParameterData(kind))
|
||
|
|
{
|
||
|
|
}
|
||
|
|
|
||
|
|
BaseFileWizardParameters::BaseFileWizardParameters(const BaseFileWizardParameters &rhs) :
|
||
|
|
m_d(rhs.m_d)
|
||
|
|
{
|
||
|
|
}
|
||
|
|
|
||
|
|
BaseFileWizardParameters &BaseFileWizardParameters::operator=(const BaseFileWizardParameters &rhs)
|
||
|
|
{
|
||
|
|
if (this != &rhs)
|
||
|
|
m_d.operator=(rhs.m_d);
|
||
|
|
return *this;
|
||
|
|
}
|
||
|
|
|
||
|
|
BaseFileWizardParameters::~BaseFileWizardParameters()
|
||
|
|
{
|
||
|
|
}
|
||
|
|
|
||
|
|
IWizard::Kind BaseFileWizardParameters::kind() const
|
||
|
|
{
|
||
|
|
return m_d->kind;
|
||
|
|
}
|
||
|
|
|
||
|
|
void BaseFileWizardParameters::setKind(IWizard::Kind k)
|
||
|
|
{
|
||
|
|
m_d->kind = k;
|
||
|
|
}
|
||
|
|
|
||
|
|
QIcon BaseFileWizardParameters::icon() const
|
||
|
|
{
|
||
|
|
return m_d->icon;
|
||
|
|
}
|
||
|
|
|
||
|
|
void BaseFileWizardParameters::setIcon(const QIcon &icon)
|
||
|
|
{
|
||
|
|
m_d->icon = icon;
|
||
|
|
}
|
||
|
|
|
||
|
|
QString BaseFileWizardParameters::description() const
|
||
|
|
{
|
||
|
|
return m_d->description;
|
||
|
|
}
|
||
|
|
|
||
|
|
void BaseFileWizardParameters::setDescription(const QString &v)
|
||
|
|
{
|
||
|
|
m_d->description = v;
|
||
|
|
}
|
||
|
|
|
||
|
|
|
||
|
|
QString BaseFileWizardParameters::name() const
|
||
|
|
{
|
||
|
|
return m_d->name;
|
||
|
|
}
|
||
|
|
|
||
|
|
void BaseFileWizardParameters::setName(const QString &v)
|
||
|
|
{
|
||
|
|
m_d->name = v;
|
||
|
|
}
|
||
|
|
|
||
|
|
|
||
|
|
QString BaseFileWizardParameters::category() const
|
||
|
|
{
|
||
|
|
return m_d->category;
|
||
|
|
}
|
||
|
|
|
||
|
|
void BaseFileWizardParameters::setCategory(const QString &v)
|
||
|
|
{
|
||
|
|
m_d->category = v;
|
||
|
|
}
|
||
|
|
|
||
|
|
QString BaseFileWizardParameters::trCategory() const
|
||
|
|
{
|
||
|
|
return m_d->trCategory;
|
||
|
|
}
|
||
|
|
|
||
|
|
void BaseFileWizardParameters::setTrCategory(const QString &v)
|
||
|
|
{
|
||
|
|
m_d->trCategory = v;
|
||
|
|
}
|
||
|
|
|
||
|
|
/* WizardEventLoop: Special event loop that runs a QWizard and terminates if the page changes.
|
||
|
|
* Synopsis:
|
||
|
|
* \code
|
||
|
|
Wizard wizard(parent);
|
||
|
|
WizardEventLoop::WizardResult wr;
|
||
|
|
do {
|
||
|
|
wr = WizardEventLoop::execWizardPage(wizard);
|
||
|
|
} while (wr == WizardEventLoop::PageChanged);
|
||
|
|
* \endcode */
|
||
|
|
|
||
|
|
class WizardEventLoop : public QEventLoop
|
||
|
|
{
|
||
|
|
Q_OBJECT
|
||
|
|
Q_DISABLE_COPY(WizardEventLoop)
|
||
|
|
WizardEventLoop(QObject *parent);
|
||
|
|
|
||
|
|
public:
|
||
|
|
enum WizardResult { Accepted, Rejected , PageChanged };
|
||
|
|
|
||
|
|
static WizardResult execWizardPage(QWizard &w);
|
||
|
|
|
||
|
|
private slots:
|
||
|
|
void pageChanged(int);
|
||
|
|
void accepted();
|
||
|
|
void rejected();
|
||
|
|
|
||
|
|
private:
|
||
|
|
WizardResult execWizardPageI();
|
||
|
|
|
||
|
|
WizardResult m_result;
|
||
|
|
};
|
||
|
|
|
||
|
|
WizardEventLoop::WizardEventLoop(QObject *parent) :
|
||
|
|
QEventLoop(parent),
|
||
|
|
m_result(Rejected)
|
||
|
|
{
|
||
|
|
}
|
||
|
|
|
||
|
|
WizardEventLoop::WizardResult WizardEventLoop::execWizardPage(QWizard &wizard)
|
||
|
|
{
|
||
|
|
/* Install ourselves on the wizard. Main trick is here to connect
|
||
|
|
* to the page changed signal and quit() on it. */
|
||
|
|
WizardEventLoop *eventLoop = wizard.findChild<WizardEventLoop *>();
|
||
|
|
if (!eventLoop) {
|
||
|
|
eventLoop = new WizardEventLoop(&wizard);
|
||
|
|
connect(&wizard, SIGNAL(currentIdChanged(int)), eventLoop, SLOT(pageChanged(int)));
|
||
|
|
connect(&wizard, SIGNAL(accepted()), eventLoop, SLOT(accepted()));
|
||
|
|
connect(&wizard, SIGNAL(rejected()), eventLoop, SLOT(rejected()));
|
||
|
|
wizard.setAttribute(Qt::WA_ShowModal, true);
|
||
|
|
wizard.show();
|
||
|
|
}
|
||
|
|
const WizardResult result = eventLoop->execWizardPageI();
|
||
|
|
// Quitting?
|
||
|
|
if (result != PageChanged)
|
||
|
|
delete eventLoop;
|
||
|
|
if (debugWizard)
|
||
|
|
qDebug() << "WizardEventLoop::runWizard" << wizard.pageIds() << " returns " << result;
|
||
|
|
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
|
||
|
|
WizardEventLoop::WizardResult WizardEventLoop::execWizardPageI()
|
||
|
|
{
|
||
|
|
m_result = Rejected;
|
||
|
|
exec(QEventLoop::DialogExec);
|
||
|
|
return m_result;
|
||
|
|
}
|
||
|
|
|
||
|
|
void WizardEventLoop::pageChanged(int /*page*/)
|
||
|
|
{
|
||
|
|
m_result = PageChanged;
|
||
|
|
quit(); // !
|
||
|
|
}
|
||
|
|
|
||
|
|
void WizardEventLoop::accepted()
|
||
|
|
{
|
||
|
|
m_result = Accepted;
|
||
|
|
quit();
|
||
|
|
}
|
||
|
|
|
||
|
|
void WizardEventLoop::rejected()
|
||
|
|
{
|
||
|
|
m_result = Rejected;
|
||
|
|
quit();
|
||
|
|
}
|
||
|
|
|
||
|
|
// ---------------- BaseFileWizardPrivate
|
||
|
|
struct BaseFileWizardPrivate {
|
||
|
|
explicit BaseFileWizardPrivate(const Core::BaseFileWizardParameters ¶meters,
|
||
|
|
Core::ICore *core);
|
||
|
|
|
||
|
|
const Core::BaseFileWizardParameters m_parameters;
|
||
|
|
QWizard *m_wizardDialog;
|
||
|
|
Core::ICore *m_core;
|
||
|
|
};
|
||
|
|
|
||
|
|
Core::BaseFileWizardPrivate::BaseFileWizardPrivate(const BaseFileWizardParameters ¶meters,
|
||
|
|
Core::ICore *core) :
|
||
|
|
m_parameters(parameters),
|
||
|
|
m_wizardDialog(0),
|
||
|
|
m_core(core)
|
||
|
|
{
|
||
|
|
}
|
||
|
|
|
||
|
|
// ---------------- Wizard
|
||
|
|
BaseFileWizard::BaseFileWizard(const BaseFileWizardParameters ¶meters,
|
||
|
|
Core::ICore *core,
|
||
|
|
QObject *parent) :
|
||
|
|
IWizard(parent),
|
||
|
|
m_d(new BaseFileWizardPrivate(parameters, core))
|
||
|
|
{
|
||
|
|
}
|
||
|
|
|
||
|
|
BaseFileWizard::~BaseFileWizard()
|
||
|
|
{
|
||
|
|
delete m_d;
|
||
|
|
}
|
||
|
|
|
||
|
|
IWizard::Kind BaseFileWizard::kind() const
|
||
|
|
{
|
||
|
|
return m_d->m_parameters.kind();
|
||
|
|
}
|
||
|
|
|
||
|
|
QIcon BaseFileWizard::icon() const
|
||
|
|
{
|
||
|
|
return m_d->m_parameters.icon();
|
||
|
|
}
|
||
|
|
|
||
|
|
QString BaseFileWizard::description() const
|
||
|
|
{
|
||
|
|
return m_d->m_parameters.description();
|
||
|
|
}
|
||
|
|
|
||
|
|
QString BaseFileWizard::name() const
|
||
|
|
{
|
||
|
|
return m_d->m_parameters.name();
|
||
|
|
}
|
||
|
|
|
||
|
|
QString BaseFileWizard::category() const
|
||
|
|
{
|
||
|
|
return m_d->m_parameters.category();
|
||
|
|
}
|
||
|
|
|
||
|
|
QString BaseFileWizard::trCategory() const
|
||
|
|
{
|
||
|
|
return m_d->m_parameters.trCategory();
|
||
|
|
}
|
||
|
|
|
||
|
|
QStringList BaseFileWizard::runWizard(const QString &path, QWidget *parent)
|
||
|
|
{
|
||
|
|
typedef QList<IFileWizardExtension*> ExtensionList;
|
||
|
|
|
||
|
|
QString errorMessage;
|
||
|
|
// Compile extension pages, purge out unused ones
|
||
|
|
ExtensionList extensions = ExtensionSystem::PluginManager::instance()->getObjects<IFileWizardExtension>();
|
||
|
|
WizardPageList allExtensionPages;
|
||
|
|
for (ExtensionList::iterator it = extensions.begin(); it != extensions.end(); ) {
|
||
|
|
const WizardPageList extensionPages = (*it)->extensionPages(this);
|
||
|
|
if (extensionPages.empty()) {
|
||
|
|
it = extensions.erase(it);
|
||
|
|
} else {
|
||
|
|
allExtensionPages += extensionPages;
|
||
|
|
++it;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
if (debugWizard)
|
||
|
|
qDebug() << Q_FUNC_INFO << path << parent << "exs" << extensions.size() << allExtensionPages.size();
|
||
|
|
|
||
|
|
QWizardPage *firstExtensionPage = 0;
|
||
|
|
if (!allExtensionPages.empty())
|
||
|
|
firstExtensionPage = allExtensionPages.front();
|
||
|
|
|
||
|
|
// Create dialog and run it. Ensure that the dialog is deleted when
|
||
|
|
// leaving the func, but not before the IFileWizardExtension::process
|
||
|
|
// has been called
|
||
|
|
const QSharedPointer<QWizard> wizard(createWizardDialog(parent, path, allExtensionPages));
|
||
|
|
GeneratedFiles files;
|
||
|
|
// Run the wizard: Call generate files on switching to the first extension
|
||
|
|
// page is OR after 'Accepted' if there are no extension pages
|
||
|
|
while (true) {
|
||
|
|
const WizardEventLoop::WizardResult wr = WizardEventLoop::execWizardPage(*wizard);
|
||
|
|
if (wr == WizardEventLoop::Rejected) {
|
||
|
|
files.clear();
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
const bool accepted = wr == WizardEventLoop::Accepted;
|
||
|
|
const bool firstExtensionPageHit = wr == WizardEventLoop::PageChanged
|
||
|
|
&& wizard->page(wizard->currentId()) == firstExtensionPage;
|
||
|
|
const bool needGenerateFiles = firstExtensionPageHit || (accepted && allExtensionPages.empty());
|
||
|
|
if (needGenerateFiles) {
|
||
|
|
QString errorMessage;
|
||
|
|
files = generateFiles(wizard.data(), &errorMessage);
|
||
|
|
if (files.empty()) {
|
||
|
|
QMessageBox::critical(0, tr("File Generation Failure"), errorMessage);
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
if (firstExtensionPageHit)
|
||
|
|
foreach(IFileWizardExtension *ex, extensions)
|
||
|
|
ex->firstExtensionPageShown(files);
|
||
|
|
if (accepted)
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
if (files.empty())
|
||
|
|
return QStringList();
|
||
|
|
// Compile result list and prompt for overwrite
|
||
|
|
QStringList result;
|
||
|
|
foreach (const GeneratedFile &generatedFile, files)
|
||
|
|
result.push_back(generatedFile.path());
|
||
|
|
|
||
|
|
switch (promptOverwrite(path, result, &errorMessage)) {
|
||
|
|
case OverwriteCanceled:
|
||
|
|
return QStringList();
|
||
|
|
case OverwriteError:
|
||
|
|
QMessageBox::critical(0, tr("Existing files"), errorMessage);
|
||
|
|
return QStringList();
|
||
|
|
case OverwriteOk:
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
|
||
|
|
// Write
|
||
|
|
foreach (const GeneratedFile &generatedFile, files) {
|
||
|
|
if (!generatedFile.write(&errorMessage)) {
|
||
|
|
QMessageBox::critical(parent, tr("File Generation Failure"), errorMessage);
|
||
|
|
return QStringList();
|
||
|
|
}
|
||
|
|
}
|
||
|
|
// Run the extensions
|
||
|
|
foreach(IFileWizardExtension *ex, extensions)
|
||
|
|
if (!ex->process(files, &errorMessage)) {
|
||
|
|
QMessageBox::critical(parent, tr("File Generation Failure"), errorMessage);
|
||
|
|
return QStringList();
|
||
|
|
}
|
||
|
|
|
||
|
|
// Post generation handler
|
||
|
|
if (!postGenerateFiles(files, &errorMessage)) {
|
||
|
|
QMessageBox::critical(0, tr("File Generation Failure"), errorMessage);
|
||
|
|
return QStringList();
|
||
|
|
}
|
||
|
|
|
||
|
|
return result;
|
||
|
|
}
|
||
|
|
|
||
|
|
QPixmap BaseFileWizard::watermark()
|
||
|
|
{
|
||
|
|
return QPixmap(QLatin1String(":/qworkbench/images/qtwatermark.png"));
|
||
|
|
}
|
||
|
|
|
||
|
|
void BaseFileWizard::setupWizard(QWizard *w)
|
||
|
|
{
|
||
|
|
w->setPixmap(QWizard::WatermarkPixmap, watermark());
|
||
|
|
}
|
||
|
|
|
||
|
|
bool BaseFileWizard::postGenerateFiles(const GeneratedFiles &l, QString *errorMessage)
|
||
|
|
{
|
||
|
|
// File mode: open the editors in file mode and ensure editor pane
|
||
|
|
const Core::GeneratedFiles::const_iterator cend = l.constEnd();
|
||
|
|
for (Core::GeneratedFiles::const_iterator it = l.constBegin(); it != cend; ++it) {
|
||
|
|
if (!m_d->m_core->editorManager()->openEditor(it->path(), it->editorKind())) {
|
||
|
|
*errorMessage = tr("Failed to open an editor for %1").arg(it->path());
|
||
|
|
return false;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
m_d->m_core->editorManager()->ensureEditorManagerVisible();
|
||
|
|
return true;
|
||
|
|
}
|
||
|
|
|
||
|
|
BaseFileWizard::OverwriteResult BaseFileWizard::promptOverwrite(const QString &location,
|
||
|
|
const QStringList &files,
|
||
|
|
QString *errorMessage) const
|
||
|
|
{
|
||
|
|
if (debugWizard)
|
||
|
|
qDebug() << Q_FUNC_INFO << location << files;
|
||
|
|
|
||
|
|
|
||
|
|
bool existingFilesFound = false;
|
||
|
|
bool oddStuffFound = false;
|
||
|
|
|
||
|
|
static const QString readOnlyMsg = tr(" [read only]");
|
||
|
|
static const QString directoryMsg = tr(" [directory]");
|
||
|
|
static const QString symLinkMsg = tr(" [symbolic link]");
|
||
|
|
|
||
|
|
// Format a file list message as ( "<file1> [readonly], <file2> [directory]").
|
||
|
|
QString fileNamesMsgPart;
|
||
|
|
foreach (const QString &fileName, files) {
|
||
|
|
const QFileInfo fi(fileName);
|
||
|
|
if (fi.exists()) {
|
||
|
|
existingFilesFound = true;
|
||
|
|
if (!fileNamesMsgPart.isEmpty())
|
||
|
|
fileNamesMsgPart += QLatin1String(", ");
|
||
|
|
fileNamesMsgPart += fi.fileName();
|
||
|
|
do {
|
||
|
|
if (fi.isDir()) {
|
||
|
|
oddStuffFound = true;
|
||
|
|
fileNamesMsgPart += directoryMsg;
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
if (fi.isSymLink()) {
|
||
|
|
oddStuffFound = true;
|
||
|
|
fileNamesMsgPart += symLinkMsg;
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
if (!fi.isWritable()) {
|
||
|
|
oddStuffFound = true;
|
||
|
|
fileNamesMsgPart += readOnlyMsg;
|
||
|
|
}
|
||
|
|
} while (false);
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
if (!existingFilesFound)
|
||
|
|
return OverwriteOk;
|
||
|
|
|
||
|
|
if (oddStuffFound) {
|
||
|
|
*errorMessage = tr("The project directory %1 contains files which cannot be overwritten:\n%2.").arg(location).arg(fileNamesMsgPart);
|
||
|
|
return OverwriteError;
|
||
|
|
}
|
||
|
|
|
||
|
|
const QString messageFormat = tr("The following files already exist in the directory %1:\n"
|
||
|
|
"%2.\nWould you like to overwrite them?");
|
||
|
|
const QString message = messageFormat.arg(location).arg(fileNamesMsgPart);
|
||
|
|
const bool yes = (QMessageBox::question(core()->mainWindow(),
|
||
|
|
tr("Existing files"), message,
|
||
|
|
QMessageBox::Yes | QMessageBox::No,
|
||
|
|
QMessageBox::No)
|
||
|
|
== QMessageBox::Yes);
|
||
|
|
return yes ? OverwriteOk : OverwriteCanceled;
|
||
|
|
}
|
||
|
|
|
||
|
|
Core::ICore *BaseFileWizard::core() const
|
||
|
|
{
|
||
|
|
return m_d->m_core;
|
||
|
|
}
|
||
|
|
|
||
|
|
QList<IWizard*> BaseFileWizard::allWizards()
|
||
|
|
{
|
||
|
|
return ExtensionSystem::PluginManager::instance()->getObjects<IWizard>();
|
||
|
|
}
|
||
|
|
|
||
|
|
// Utility to find all registered wizards of a certain kind
|
||
|
|
|
||
|
|
class WizardKindPredicate {
|
||
|
|
public:
|
||
|
|
WizardKindPredicate(IWizard::Kind kind) : m_kind(kind) {}
|
||
|
|
bool operator()(const IWizard &w) const { return w.kind() == m_kind; }
|
||
|
|
private:
|
||
|
|
const IWizard::Kind m_kind;
|
||
|
|
};
|
||
|
|
|
||
|
|
QList<IWizard*> BaseFileWizard::findWizardsOfKind(Kind kind)
|
||
|
|
{
|
||
|
|
return findWizards(WizardKindPredicate(kind));
|
||
|
|
}
|
||
|
|
|
||
|
|
QString BaseFileWizard::buildFileName(const QString &path,
|
||
|
|
const QString &baseName,
|
||
|
|
const QString &extension)
|
||
|
|
{
|
||
|
|
QString rc = path;
|
||
|
|
if (!rc.isEmpty() && !rc.endsWith(QDir::separator()))
|
||
|
|
rc += QDir::separator();
|
||
|
|
rc += baseName;
|
||
|
|
// Add extension unless user specified something else
|
||
|
|
const QChar dot = QLatin1Char('.');
|
||
|
|
if (!extension.isEmpty() && !baseName.contains(dot)) {
|
||
|
|
if (!extension.startsWith(dot))
|
||
|
|
rc += dot;
|
||
|
|
rc += extension;
|
||
|
|
}
|
||
|
|
if (debugWizard)
|
||
|
|
qDebug() << Q_FUNC_INFO << rc;
|
||
|
|
return rc;
|
||
|
|
}
|
||
|
|
|
||
|
|
QString BaseFileWizard::preferredSuffix(const QString &mimeType) const
|
||
|
|
{
|
||
|
|
const QString rc = m_d->m_core->mimeDatabase()->preferredSuffixByType(mimeType);
|
||
|
|
if (rc.isEmpty())
|
||
|
|
qWarning("%s: WARNING: Unable to find a preferred suffix for %s.",
|
||
|
|
Q_FUNC_INFO, mimeType.toUtf8().constData());
|
||
|
|
return rc;
|
||
|
|
}
|
||
|
|
|
||
|
|
// ------------- StandardFileWizard(
|
||
|
|
|
||
|
|
StandardFileWizard::StandardFileWizard(const BaseFileWizardParameters ¶meters,
|
||
|
|
Core::ICore *core,
|
||
|
|
QObject *parent) :
|
||
|
|
BaseFileWizard(parameters, core, parent)
|
||
|
|
{
|
||
|
|
}
|
||
|
|
|
||
|
|
QWizard *StandardFileWizard::createWizardDialog(QWidget *parent,
|
||
|
|
const QString &defaultPath,
|
||
|
|
const WizardPageList &extensionPages) const
|
||
|
|
{
|
||
|
|
Core::Utils::FileWizardDialog *standardWizardDialog = new Core::Utils::FileWizardDialog(parent);
|
||
|
|
standardWizardDialog->setWindowTitle(tr("New %1").arg(name()));
|
||
|
|
setupWizard(standardWizardDialog);
|
||
|
|
standardWizardDialog->setPath(defaultPath);
|
||
|
|
foreach (QWizardPage *p, extensionPages)
|
||
|
|
standardWizardDialog->addPage(p);
|
||
|
|
return standardWizardDialog;
|
||
|
|
}
|
||
|
|
|
||
|
|
GeneratedFiles StandardFileWizard::generateFiles(const QWizard *w,
|
||
|
|
QString *errorMessage) const
|
||
|
|
{
|
||
|
|
const Core::Utils::FileWizardDialog *standardWizardDialog = qobject_cast<const Core::Utils::FileWizardDialog *>(w);
|
||
|
|
return generateFilesFromPath(standardWizardDialog->path(),
|
||
|
|
standardWizardDialog->name(),
|
||
|
|
errorMessage);
|
||
|
|
}
|
||
|
|
|
||
|
|
|
||
|
|
} // namespace Core
|
||
|
|
#include "basefilewizard.moc"
|