#pragma once

#include <QApplication>
#include <memory>
#include <QDebug>
#include <QFlag>
#include <QIcon>
#include <QDateTime>
#include <updater/GoUpdate.h>

#include <BaseInstance.h>

class LaunchController;
class LocalPeer;
class InstanceWindow;
class MainWindow;
class SetupWizard;
class FolderInstanceProvider;
class GenericPageProvider;
class QFile;
class HttpMetaCache;
class SettingsObject;
class InstanceList;
class MojangAccountList;
class IconList;
class QNetworkAccessManager;
class JavaInstallList;
class UpdateChecker;
class BaseProfilerFactory;
class BaseDetachedToolFactory;
class TranslationsModel;
class ITheme;
class MCEditTool;
class GAnalytics;

#if defined(MMC)
#undef MMC
#endif
#define MMC (static_cast<MultiMC *>(QCoreApplication::instance()))

class MultiMC : public QApplication
{
	// friends for the purpose of limiting access to deprecated stuff
	Q_OBJECT
public:
	enum Status
	{
		StartingUp,
		Failed,
		Succeeded,
		Initialized
	};

public:
	MultiMC(int &argc, char **argv);
	virtual ~MultiMC();

	GAnalytics *analytics() const
	{
		return m_analytics;
	}

	std::shared_ptr<SettingsObject> settings() const
	{
		return m_settings;
	}

	std::shared_ptr<GenericPageProvider> globalSettingsPages() const
	{
		return m_globalSettingsProvider;
	}

	qint64 timeSinceStart() const
	{
		return startTime.msecsTo(QDateTime::currentDateTime());
	}

	QIcon getThemedIcon(const QString& name);

	void setIconTheme(const QString& name);

	std::vector<ITheme *> getValidApplicationThemes();

	void setApplicationTheme(const QString& name, bool initial);

	// DownloadUpdateTask
	std::shared_ptr<UpdateChecker> updateChecker()
	{
		return m_updateChecker;
	}

	std::shared_ptr<TranslationsModel> translations();

	std::shared_ptr<JavaInstallList> javalist();

	std::shared_ptr<InstanceList> instances() const
	{
		return m_instances;
	}

	FolderInstanceProvider * folderProvider() const
	{
		return m_instanceFolder;
	}

	std::shared_ptr<IconList> icons() const
	{
		return m_icons;
	}

	MCEditTool *mcedit() const
	{
		return m_mcedit.get();
	}

	std::shared_ptr<MojangAccountList> accounts() const
	{
		return m_accounts;
	}

	Status status() const
	{
		return m_status;
	}

	const QMap<QString, std::shared_ptr<BaseProfilerFactory>> &profilers() const
	{
		return m_profilers;
	}

	/// this is the root of the 'installation'. Used for automatic updates
	const QString &root()
	{
		return m_rootPath;
	}

	/*!
	 * Opens a json file using either a system default editor, or, if not empty, the editor
	 * specified in the settings
	 */
	bool openJsonEditor(const QString &filename);

	InstanceWindow *showInstanceWindow(InstancePtr instance, QString page = QString());
	MainWindow *showMainWindow(bool minimized = false);

	void updateIsRunning(bool running);
	bool updatesAreAllowed();

signals:
	void updateAllowedChanged(bool status);

public slots:
	bool launch(InstancePtr instance, bool online = true, BaseProfilerFactory *profiler = nullptr);
	bool kill(InstancePtr instance);

private slots:
	void on_windowClose();
	void messageReceived(const QString & message);
	void controllerSucceeded();
	void controllerFailed(const QString & error);
	void analyticsSettingChanged(const Setting &setting, QVariant value);
	void setupWizardFinished(int status);

private:
	bool createSetupWizard();
	void performMainStartupAction();

	// sets the fatal error message and m_status to Failed.
	void showFatalErrorMessage(const QString & title, const QString & content);

private:
	void addRunningInstance();
	void subRunningInstance();
	bool shouldExitNow() const;

private:
	QDateTime startTime;

	std::shared_ptr<SettingsObject> m_settings;
	std::shared_ptr<InstanceList> m_instances;
	FolderInstanceProvider * m_instanceFolder = nullptr;
	std::shared_ptr<IconList> m_icons;
	std::shared_ptr<UpdateChecker> m_updateChecker;
	std::shared_ptr<MojangAccountList> m_accounts;
	std::shared_ptr<JavaInstallList> m_javalist;
	std::shared_ptr<TranslationsModel> m_translations;
	std::shared_ptr<GenericPageProvider> m_globalSettingsProvider;
	std::map<QString, std::unique_ptr<ITheme>> m_themes;
	std::unique_ptr<MCEditTool> m_mcedit;

	QMap<QString, std::shared_ptr<BaseProfilerFactory>> m_profilers;

	QString m_rootPath;
	Status m_status = MultiMC::StartingUp;

#if defined Q_OS_WIN32
	// used on Windows to attach the standard IO streams
	bool consoleAttached = false;
#endif

	// FIXME: attach to instances instead.
	struct InstanceXtras
	{
		InstanceWindow * window = nullptr;
		shared_qobject_ptr<LaunchController> controller;
	};
	std::map<QString, InstanceXtras> m_instanceExtras;

	// main state variables
	size_t m_openWindows = 0;
	size_t m_runningInstances = 0;
	bool m_updateRunning = false;

	// main window, if any
	MainWindow * m_mainWindow = nullptr;

	// peer MultiMC instance connector - used to implement single instance MultiMC and signalling
	LocalPeer * m_peerInstance = nullptr;

	GAnalytics * m_analytics = nullptr;
	SetupWizard * m_setupWizard = nullptr;
public:
	QString m_instanceIdToLaunch;
	bool m_liveCheck = false;
	std::unique_ptr<QFile> logFile;
};