diff --git a/src/plugins/coreplugin/coreplugin.cpp b/src/plugins/coreplugin/coreplugin.cpp index ffa050e5a056e3e35579d00390e7df4a61e196fd..e4ce23df18a9cf9960937761846b1d02c4759ba2 100644 --- a/src/plugins/coreplugin/coreplugin.cpp +++ b/src/plugins/coreplugin/coreplugin.cpp @@ -185,6 +185,8 @@ bool CorePlugin::initialize(const QStringList &arguments, QString *errorMessage) InfoBar::initializeGloballySuppressed(); } + IWizardFactory::initialize(); + // Make sure we respect the process's umask when creating new files SaveFile::initializeUmask(); diff --git a/src/plugins/coreplugin/icore.cpp b/src/plugins/coreplugin/icore.cpp index dce4a74a21f8063927234673f04632eb78446312..3550d827bdde0080703fddbd6bc8eba62e482f33 100644 --- a/src/plugins/coreplugin/icore.cpp +++ b/src/plugins/coreplugin/icore.cpp @@ -533,11 +533,6 @@ void ICore::openFiles(const QStringList &arguments, ICore::OpenFilesFlags flags) m_mainwindow->openFiles(arguments, flags); } -void ICore::emitNewItemsDialogRequested() -{ - emit m_instance->newItemsDialogRequested(); -} - void ICore::saveSettings() { emit m_instance->saveSettingsRequested(); diff --git a/src/plugins/coreplugin/icore.h b/src/plugins/coreplugin/icore.h index ffb45dcd7c43a014b169207afaa8b6f864359122..61df88c61b202f6c5d24c73ac9e141b5c4b29039 100644 --- a/src/plugins/coreplugin/icore.h +++ b/src/plugins/coreplugin/icore.h @@ -123,15 +123,12 @@ public: }; static void openFiles(const QStringList &fileNames, OpenFilesFlags flags = None); - static void emitNewItemsDialogRequested(); - public slots: static void saveSettings(); signals: void coreAboutToOpen(); void coreOpened(); - void newItemsDialogRequested(); void newItemDialogRunningChanged(); void saveSettingsRequested(); void optionsDialogRequested(); diff --git a/src/plugins/coreplugin/iwizardfactory.cpp b/src/plugins/coreplugin/iwizardfactory.cpp index 4262b4abc53b4a875afb39801a88f8b7b50f643f..0feac766a19cdd0b36e849f19bd11528ea6446ea 100644 --- a/src/plugins/coreplugin/iwizardfactory.cpp +++ b/src/plugins/coreplugin/iwizardfactory.cpp @@ -147,16 +147,18 @@ using namespace Core; + namespace { static QList<IFeatureProvider *> s_providerList; +QList<IWizardFactory *> s_allFactories; +QList<IWizardFactory::FactoryCreator> s_factoryCreators; +bool s_areFactoriesLoaded = false; } /* A utility to find all wizards supporting a view mode and matching a predicate */ template <class Predicate> QList<IWizardFactory*> findWizardFactories(Predicate predicate) { - // Hack: Trigger delayed creation of wizards - ICore::emitNewItemsDialogRequested(); // Filter all wizards const QList<IWizardFactory*> allFactories = IWizardFactory::allWizardFactories(); QList<IWizardFactory*> rc; @@ -169,9 +171,33 @@ template <class Predicate> QList<IWizardFactory*> IWizardFactory::allWizardFactories() { - // Hack: Trigger delayed creation of wizards - ICore::emitNewItemsDialogRequested(); - return ExtensionSystem::PluginManager::getObjects<IWizardFactory>(); + if (!s_areFactoriesLoaded) { + QTC_ASSERT(s_allFactories.isEmpty(), return s_allFactories); + + s_areFactoriesLoaded = true; + + QHash<Id, IWizardFactory *> sanityCheck; + foreach (const FactoryCreator &fc, s_factoryCreators) { + QList<IWizardFactory *> tmp = fc(); + foreach (IWizardFactory *newFactory, tmp) { + QTC_ASSERT(newFactory, continue); + IWizardFactory *existingFactory = sanityCheck.value(newFactory->id()); + + QTC_ASSERT(existingFactory != newFactory, continue); + if (existingFactory) { + qWarning("%s", qPrintable(tr("Factory with id=\"%1\" already registered. Deleting.") + .arg(existingFactory->id().toString()))); + delete newFactory; + continue; + } + + sanityCheck.insert(newFactory->id(), newFactory); + s_allFactories << newFactory; + } + } + } + + return s_allFactories; } // Utility to find all registered wizards of a certain kind @@ -211,6 +237,11 @@ QStringList IWizardFactory::supportedPlatforms() const return stringList; } +void IWizardFactory::registerFactoryCreator(const IWizardFactory::FactoryCreator &creator) +{ + s_factoryCreators << creator; +} + QStringList IWizardFactory::allAvailablePlatforms() { QStringList platforms; @@ -257,3 +288,9 @@ FeatureSet IWizardFactory::pluginFeatures() const } return plugins; } + +void IWizardFactory::initialize() +{ + connect(ICore::instance(), &ICore::coreAboutToClose, + ICore::instance(), []() { qDeleteAll(s_allFactories); s_allFactories.clear(); }); +} diff --git a/src/plugins/coreplugin/iwizardfactory.h b/src/plugins/coreplugin/iwizardfactory.h index df5a4e81399a8c19dcd059167cac5f075e95bf26..0d52a0e54a382c6397da25e3776b70f4e339478e 100644 --- a/src/plugins/coreplugin/iwizardfactory.h +++ b/src/plugins/coreplugin/iwizardfactory.h @@ -38,6 +38,8 @@ #include <QObject> #include <QString> +#include <functional> + namespace Core { namespace Internal { class CorePlugin; } @@ -87,6 +89,9 @@ public: bool isAvailable(const QString &platformName) const; QStringList supportedPlatforms() const; + typedef std::function<QList<IWizardFactory *>()> FactoryCreator; + static void registerFactoryCreator(const FactoryCreator &creator); + // Utility to find all registered wizards static QList<IWizardFactory*> allWizardFactories(); // Utility to find all registered wizards of a certain kind @@ -100,6 +105,7 @@ protected: FeatureSet pluginFeatures() const; private: + static void initialize(); static void destroyFeatureProvider(); IWizardFactory::WizardKind m_kind; diff --git a/src/plugins/designer/formeditorplugin.cpp b/src/plugins/designer/formeditorplugin.cpp index 58bade244aa80253dd2a2927ea0b6c146b6ab173..97a3ec314dddd0fddfff6ff5def37ba23fe5ac59 100644 --- a/src/plugins/designer/formeditorplugin.cpp +++ b/src/plugins/designer/formeditorplugin.cpp @@ -85,7 +85,21 @@ bool FormEditorPlugin::initialize(const QStringList &arguments, QString *error) { Q_UNUSED(arguments) - initializeTemplates(); +#ifdef CPP_ENABLED + IWizardFactory::registerFactoryCreator( + []() -> QList<IWizardFactory *> { + IWizardFactory *wizard = new FormClassWizard; + wizard->setWizardKind(IWizardFactory::FileWizard); + wizard->setCategory(QLatin1String(Core::Constants::WIZARD_CATEGORY_QT)); + wizard->setDisplayCategory(QCoreApplication::translate("Core", Core::Constants::WIZARD_TR_CATEGORY_QT)); + wizard->setDisplayName(tr("Qt Designer Form Class")); + wizard->setId("C.FormClass"); + wizard->setDescription(tr("Creates a Qt Designer form along with a matching class (C++ header and source file) " + "for implementation purposes. You can add the form and class to an existing Qt Widget Project.")); + + return QList<IWizardFactory *>() << wizard; + }); +#endif ProjectExplorer::JsonWizardFactory::registerPageFactory(new Internal::FormPageFactory); addAutoReleasedObject(new FormEditorFactory); @@ -130,21 +144,6 @@ void FormEditorPlugin::extensionsInitialized() // //////////////////////////////////////////////////// -void FormEditorPlugin::initializeTemplates() -{ -#ifdef CPP_ENABLED - IWizardFactory *wizard = new FormClassWizard; - wizard->setWizardKind(IWizardFactory::FileWizard); - wizard->setCategory(QLatin1String(Core::Constants::WIZARD_CATEGORY_QT)); - wizard->setDisplayCategory(QCoreApplication::translate("Core", Core::Constants::WIZARD_TR_CATEGORY_QT)); - wizard->setDisplayName(tr("Qt Designer Form Class")); - wizard->setId("C.FormClass"); - wizard->setDescription(tr("Creates a Qt Designer form along with a matching class (C++ header and source file) " - "for implementation purposes. You can add the form and class to an existing Qt Widget Project.")); - addAutoReleasedObject(wizard); -#endif -} - // Find out current existing editor file static QString currentFile() { diff --git a/src/plugins/genericprojectmanager/genericprojectplugin.cpp b/src/plugins/genericprojectmanager/genericprojectplugin.cpp index 2b784b2434687b059cffa7dc2f3ba35ae59675f1..2a595b4d26a63353025ce2f91625bb22475215dd 100644 --- a/src/plugins/genericprojectmanager/genericprojectplugin.cpp +++ b/src/plugins/genericprojectmanager/genericprojectplugin.cpp @@ -70,9 +70,10 @@ bool GenericProjectPlugin::initialize(const QStringList &, QString *errorMessage addAutoReleasedObject(new Manager); addAutoReleasedObject(new ProjectFilesFactory); addAutoReleasedObject(new GenericMakeStepFactory); - addAutoReleasedObject(new GenericProjectWizard); addAutoReleasedObject(new GenericBuildConfigurationFactory); + IWizardFactory::registerFactoryCreator([]() { return QList<IWizardFactory *>() << new GenericProjectWizard; }); + ActionContainer *mproject = ActionManager::actionContainer(ProjectExplorer::Constants::M_PROJECTCONTEXT); diff --git a/src/plugins/projectexplorer/customwizard/customwizard.cpp b/src/plugins/projectexplorer/customwizard/customwizard.cpp index bccd1d35d7fb7fbbc49fe58ed81d0c594989fbea..2336f57f923ee812ebf0abbe559e6f927b916399 100644 --- a/src/plugins/projectexplorer/customwizard/customwizard.cpp +++ b/src/plugins/projectexplorer/customwizard/customwizard.cpp @@ -356,9 +356,9 @@ CustomWizard *CustomWizard::createWizard(const CustomProjectWizard::CustomWizard containing valid configuration files and parse them into wizards. */ -QList<CustomWizard*> CustomWizard::createWizards() +QList<Core::IWizardFactory *> CustomWizard::createWizards() { - QList<CustomWizard*> rc; + QList<Core::IWizardFactory *> rc; QString errorMessage; QString verboseLog; const QString templateDirName = Core::ICore::resourcePath() + diff --git a/src/plugins/projectexplorer/customwizard/customwizard.h b/src/plugins/projectexplorer/customwizard/customwizard.h index f31b956b1fa096009df8f2dadc2ae6c9b23917c9..6974ad9be2228a79a8349693e14265973b4e364b 100644 --- a/src/plugins/projectexplorer/customwizard/customwizard.h +++ b/src/plugins/projectexplorer/customwizard/customwizard.h @@ -102,7 +102,7 @@ public: // Create all wizards. As other plugins might register factories for derived // classes, call it in extensionsInitialized(). - static QList<CustomWizard*> createWizards(); + static QList<IWizardFactory *> createWizards(); static void setVerbose(int); static int verbose(); diff --git a/src/plugins/projectexplorer/jsonwizard/jsonwizardfactory.cpp b/src/plugins/projectexplorer/jsonwizard/jsonwizardfactory.cpp index e889bd040a1062ba798567b6fdaa9899844e77d6..ce23bc51f8bff36fbf77c031008bec5a3c2f6a59 100644 --- a/src/plugins/projectexplorer/jsonwizard/jsonwizardfactory.cpp +++ b/src/plugins/projectexplorer/jsonwizard/jsonwizardfactory.cpp @@ -205,13 +205,13 @@ static JsonWizardFactory::Page parsePage(const QVariant &value, QString *errorMe return p; } -QList<JsonWizardFactory *> JsonWizardFactory::createWizardFactories() +QList<Core::IWizardFactory *> JsonWizardFactory::createWizardFactories() { QString errorMessage; QString verboseLog; const QString wizardFileName = QLatin1String(WIZARD_FILE); - QList <JsonWizardFactory *> result; + QList <Core::IWizardFactory *> result; foreach (const Utils::FileName &path, searchPaths()) { if (path.isEmpty()) continue; diff --git a/src/plugins/projectexplorer/jsonwizard/jsonwizardfactory.h b/src/plugins/projectexplorer/jsonwizard/jsonwizardfactory.h index ff7389d088d9df6b199050fc21aa0427f69fa744..5098930bfb9422df8ccd94c31b8b31b3d38746de 100644 --- a/src/plugins/projectexplorer/jsonwizard/jsonwizardfactory.h +++ b/src/plugins/projectexplorer/jsonwizard/jsonwizardfactory.h @@ -95,7 +95,7 @@ public: private: // Create all wizards. As other plugins might register factories for derived // classes. Called when the new file dialog is shown for the first time. - static QList<JsonWizardFactory *> createWizardFactories(); + static QList<IWizardFactory *> createWizardFactories(); static JsonWizardFactory *createWizardFactory(const QVariantMap &data, const QDir &baseDir, QString *errorMessage); static QList<Utils::FileName> &searchPaths(); diff --git a/src/plugins/projectexplorer/projectexplorer.cpp b/src/plugins/projectexplorer/projectexplorer.cpp index 609b24bd56d367f278bbe3f7f146c02352dda4d3..18e3e63cfea91c2cb878ef917a102fa4877640d7 100644 --- a/src/plugins/projectexplorer/projectexplorer.cpp +++ b/src/plugins/projectexplorer/projectexplorer.cpp @@ -284,7 +284,6 @@ public: void slotUpdateRunActions(); void currentModeChanged(Core::IMode *mode, Core::IMode *oldMode); - void loadCustomWizards(); void updateWelcomePage(); @@ -490,8 +489,12 @@ bool ProjectExplorerPlugin::initialize(const QStringList &arguments, QString *er addAutoReleasedObject(new TaskHub); - connect(ICore::instance(), &ICore::newItemsDialogRequested, - dd, &ProjectExplorerPluginPrivate::loadCustomWizards); + IWizardFactory::registerFactoryCreator([]() -> QList<IWizardFactory *> { + QList<IWizardFactory *> result; + result << CustomWizard::createWizards(); + result << JsonWizardFactory::createWizardFactories(); + return result; + }); dd->m_welcomePage = new ProjectWelcomePage; connect(dd->m_welcomePage, &ProjectWelcomePage::manageSessions, @@ -1504,20 +1507,6 @@ void ProjectExplorerPlugin::extensionsInitialized() dd->m_kitManager->restoreKits(); } -void ProjectExplorerPluginPrivate::loadCustomWizards() -{ - // Add custom wizards, for which other plugins might have registered - // class factories - static bool firstTime = true; - if (firstTime) { - firstTime = false; - foreach (IWizardFactory *cpw, CustomWizard::createWizards()) - m_instance->addAutoReleasedObject(cpw); - foreach (IWizardFactory *cpw, JsonWizardFactory::createWizardFactories()) - m_instance->addAutoReleasedObject(cpw); - } -} - void ProjectExplorerPluginPrivate::updateRunWithoutDeployMenu() { m_runWithoutDeployAction->setVisible(m_projectExplorerSettings.deployBeforeRun); diff --git a/src/plugins/qmakeprojectmanager/qmakeprojectmanagerplugin.cpp b/src/plugins/qmakeprojectmanager/qmakeprojectmanagerplugin.cpp index c402e2f2f78e1bc8f764f1461820d55dd61ed6de..9500d0d6d9dd912ffccfe316164dad51060ab9c4 100644 --- a/src/plugins/qmakeprojectmanager/qmakeprojectmanagerplugin.cpp +++ b/src/plugins/qmakeprojectmanager/qmakeprojectmanagerplugin.cpp @@ -106,11 +106,12 @@ bool QmakeProjectManagerPlugin::initialize(const QStringList &arguments, QString ProjectExplorer::KitManager::registerKitInformation(new QmakeKitInformation); - addAutoReleasedObject(new SubdirsProjectWizard); - addAutoReleasedObject(new GuiAppWizard); - addAutoReleasedObject(new LibraryWizard); - addAutoReleasedObject(new TestWizard); - addAutoReleasedObject(new CustomWidgetWizard); + IWizardFactory::registerFactoryCreator([]() { + QList<IWizardFactory *> result; + result << new SubdirsProjectWizard << new GuiAppWizard << new LibraryWizard + << new TestWizard << new CustomWidgetWizard; + return result; + }); addAutoReleasedObject(new CustomWizardMetaFactory<CustomQmakeProjectWizard> (QLatin1String("qmakeproject"), IWizardFactory::ProjectWizard));