123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263 |
- /*
- * Copyright (c) Contributors to the Open 3D Engine Project.
- * For complete copyright and license terms please see the LICENSE at the root of this distribution.
- *
- * SPDX-License-Identifier: Apache-2.0 OR MIT
- *
- */
- #include <Application.h>
- #include <ProjectUtils.h>
- #include <AzCore/IO/FileIO.h>
- #include <AzCore/Utils/Utils.h>
- #include <AzCore/Settings/SettingsRegistryMergeUtils.h>
- #include <AzFramework/Logging/LoggingComponent.h>
- #include <AzQtComponents/Utilities/HandleDpiAwareness.h>
- #include <AzQtComponents/Components/StyleManager.h>
- #include <AzQtComponents/Components/WindowDecorationWrapper.h>
- #include <ProjectManager_Traits_Platform.h>
- #include <QApplication>
- #include <QDir>
- #include <QMessageBox>
- #include <QInputDialog>
- #include <QIcon>
- namespace O3DE::ProjectManager
- {
- Application::~Application()
- {
- TearDown();
- }
- bool Application::Init(bool interactive, AZStd::unique_ptr<PythonBindings> pythonBindings)
- {
- constexpr const char* applicationName { "O3DE" };
- QApplication::setOrganizationName(applicationName);
- QApplication::setOrganizationDomain("o3de.org");
- QCoreApplication::setApplicationName(applicationName);
- QCoreApplication::setApplicationVersion("1.0");
- // Use the LogComponent for non-dev logging log
- RegisterComponentDescriptor(AzFramework::LogComponent::CreateDescriptor());
- // set the log alias to .o3de/Logs instead of the default user/logs
- AZ::IO::FixedMaxPath path = AZ::Utils::GetO3deLogsDirectory();
- // DevWriteStorage is where the event log is written during development
- m_settingsRegistry->Set(AZ::SettingsRegistryMergeUtils::FilePathKey_DevWriteStorage, path.LexicallyNormal().Native());
- // Save event logs to .o3de/Logs/eventlogger/EventLogO3DE.azsl
- m_settingsRegistry->Set(AZ::SettingsRegistryMergeUtils::BuildTargetNameKey, applicationName);
- Start(AzFramework::Application::Descriptor());
- QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
- QCoreApplication::setAttribute(Qt::AA_UseHighDpiPixmaps);
- QCoreApplication::setAttribute(Qt::AA_DontCreateNativeWidgetSiblings);
- QGuiApplication::setHighDpiScaleFactorRoundingPolicy(Qt::HighDpiScaleFactorRoundingPolicy::PassThrough);
- AzQtComponents::Utilities::HandleDpiAwareness(AzQtComponents::Utilities::SystemDpiAware);
- // Create the actual Qt Application - this needs to happen before using QMessageBox
- m_app.reset(new QApplication(*GetArgC(), *GetArgV()));
- if(!InitLog(applicationName))
- {
- AZ_Warning("ProjectManager", false, "Failed to init logging");
- }
- // Set window icon after QGuiApplication is created otherwise QPixmap for the icon fails to intialize
- QApplication::setWindowIcon(QIcon(":/ProjectManager-Icon.ico"));
- // unit tests may provide custom python bindings
- m_pythonBindings = pythonBindings ? AZStd::move(pythonBindings) : AZStd::make_unique<PythonBindings>(GetEngineRoot());
- // If the first attempt of starting python failed, then attempt to bootstrap python by
- // calling the get python script
- if (!m_pythonBindings->PythonStarted())
- {
- auto getPythonResult = ProjectUtils::RunGetPythonScript(GetEngineRoot());
- if (getPythonResult.IsSuccess())
- {
- // If the bootstrap for python was successful, then attempt to start python again
- m_pythonBindings->StartPython();
- }
- }
- if (!m_pythonBindings->PythonStarted())
- {
- if (interactive)
- {
- QMessageBox::critical(nullptr, QObject::tr("Failed to start Python"),
- QObject::tr("This tool requires an O3DE engine with a Python runtime, "
- "but was unable to automatically install O3DE's built-in Python."
- "You can troubleshoot this issue by trying to manually install O3DE's built-in "
- "Python by running the '%1' script.")
- .arg(GetPythonScriptPath));
- }
- return false;
- }
- m_settings = AZStd::make_unique<Settings>();
- if (!RegisterEngine(interactive))
- {
- return false;
- }
- const AZ::CommandLine* commandLine = GetCommandLine();
- AZ_Assert(commandLine, "Failed to get command line");
- ProjectManagerScreen startScreen = ProjectManagerScreen::Projects;
- if (size_t screenSwitchCount = commandLine->GetNumSwitchValues("screen"); screenSwitchCount > 0)
- {
- QString screenOption = commandLine->GetSwitchValue("screen", screenSwitchCount - 1).c_str();
- ProjectManagerScreen screen = ProjectUtils::GetProjectManagerScreen(screenOption);
- if (screen != ProjectManagerScreen::Invalid)
- {
- startScreen = screen;
- }
- }
- AZ::IO::FixedMaxPath projectPath;
- if (size_t projectSwitchCount = commandLine->GetNumSwitchValues("project-path"); projectSwitchCount > 0)
- {
- projectPath = commandLine->GetSwitchValue("project-path", projectSwitchCount - 1).c_str();
- }
- m_mainWindow.reset(new ProjectManagerWindow(nullptr, projectPath, startScreen));
- return true;
- }
- bool Application::InitLog(const char* logName)
- {
- if (!m_entity)
- {
- // override the log alias to the O3de Logs directory instead of the default project user/Logs folder
- AZ::IO::FixedMaxPath path = AZ::Utils::GetO3deLogsDirectory();
- AZ::IO::FileIOBase* fileIO = AZ::IO::FileIOBase::GetInstance();
- AZ_Assert(fileIO, "Failed to get FileIOBase instance");
- fileIO->SetAlias("@log@", path.LexicallyNormal().Native().c_str());
- // this entity exists because we need a home for LogComponent
- // and cannot use the system entity because we need to be able to call SetLogFileBaseName
- // so the log will be named O3DE.log
- m_entity = aznew AZ::Entity("Application Entity");
- if (m_entity)
- {
- AzFramework::LogComponent* logger = aznew AzFramework::LogComponent();
- AZ_Assert(logger, "Failed to create LogComponent");
- logger->SetLogFileBaseName(logName);
- m_entity->AddComponent(logger);
- m_entity->Init();
- m_entity->Activate();
- }
- }
- return m_entity != nullptr;
- }
- bool Application::RegisterEngine(bool interactive)
- {
- auto engineInfoOutcome = m_pythonBindings->GetEngineInfo();
- if (!engineInfoOutcome)
- {
- if (interactive)
- {
- QMessageBox::critical(nullptr,
- QObject::tr("Failed to get engine info"),
- QObject::tr("A valid engine.json could not be found or loaded. "
- "Please verify a valid engine.json file exists in %1")
- .arg(GetEngineRoot()));
- }
- AZ_Error("Project Manager", false, "Failed to get engine info");
- return false;
- }
- EngineInfo engineInfo = engineInfoOutcome.GetValue();
- if (engineInfo.m_registered)
- {
- return true;
- }
- // We no longer force registration because we no longer require that only one engine can
- // be registered with each engine name
- constexpr bool forceRegistration = false;
- auto registerOutcome = m_pythonBindings->SetEngineInfo(engineInfo, forceRegistration);
- if (!registerOutcome)
- {
- if (interactive)
- {
- ProjectUtils::DisplayDetailedError(QObject::tr("Failed to register engine"), registerOutcome);
- }
-
- AZ_Error("Project Manager", false, "Failed to register engine %s : %s",
- engineInfo.m_path.toUtf8().constData(), registerOutcome.GetError().first.c_str());
- return false;
- }
- return true;
- }
- void Application::TearDown()
- {
- if (m_entity)
- {
- m_entity->Deactivate();
- delete m_entity;
- m_entity = nullptr;
- }
- m_pythonBindings.reset();
- m_mainWindow.reset();
- m_app.reset();
- }
- bool Application::Run()
- {
- // Set up the Style Manager
- AzQtComponents::StyleManager styleManager(qApp);
- styleManager.initialize(qApp, GetEngineRoot());
- // setup stylesheets and hot reloading
- AZ::IO::FixedMaxPath engineRoot(GetEngineRoot());
- QDir rootDir(engineRoot.c_str());
- const auto pathOnDisk = rootDir.absoluteFilePath("Code/Tools/ProjectManager/Resources");
- const auto qrcPath = QStringLiteral(":/ProjectManager/style");
- AzQtComponents::StyleManager::addSearchPaths("style", pathOnDisk, qrcPath, engineRoot);
- // set stylesheet after creating the main window or their styles won't get updated
- AzQtComponents::StyleManager::setStyleSheet(m_mainWindow.data(), QStringLiteral("style:ProjectManager.qss"));
- // the decoration wrapper is intended to remember window positioning and sizing
- #if AZ_TRAIT_PROJECT_MANAGER_CUSTOM_TITLEBAR
- auto wrapper = new AzQtComponents::WindowDecorationWrapper();
- #else
- auto wrapper = new AzQtComponents::WindowDecorationWrapper(AzQtComponents::WindowDecorationWrapper::OptionDisabled);
- #endif
- wrapper->setGuest(m_mainWindow.data());
- // show the main window here to apply the stylesheet before restoring geometry or we
- // can end up with empty white space at the bottom of the window until the frame is resized again
- m_mainWindow->show();
- wrapper->enableSaveRestoreGeometry("O3DE", "ProjectManager", "mainWindowGeometry");
- wrapper->showFromSettings();
- qApp->setQuitOnLastWindowClosed(true);
- // Run the application
- return qApp->exec();
- }
- }
|