329 lines
		
	
	
	
		
			9.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			329 lines
		
	
	
	
		
			9.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
This file is part of Telegram Desktop,
 | 
						|
the official desktop version of Telegram messaging app, see https://telegram.org
 | 
						|
 | 
						|
Telegram Desktop 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.
 | 
						|
 | 
						|
It 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.
 | 
						|
 | 
						|
In addition, as a special exception, the copyright holders give permission
 | 
						|
to link the code of portions of this program with the OpenSSL library.
 | 
						|
 | 
						|
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
 | 
						|
Copyright (c) 2014-2015 John Preston, https://desktop.telegram.org
 | 
						|
*/
 | 
						|
#include "stdafx.h"
 | 
						|
 | 
						|
#include "window.h"
 | 
						|
#include "mainwidget.h"
 | 
						|
 | 
						|
#include "layerwidget.h"
 | 
						|
#include "lang.h"
 | 
						|
 | 
						|
namespace App {
 | 
						|
 | 
						|
	void sendBotCommand(const QString &cmd, MsgId replyTo) {
 | 
						|
		if (MainWidget *m = main()) m->sendBotCommand(cmd, replyTo);
 | 
						|
	}
 | 
						|
 | 
						|
	bool insertBotCommand(const QString &cmd, bool specialGif) {
 | 
						|
		if (MainWidget *m = main()) return m->insertBotCommand(cmd, specialGif);
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	void searchByHashtag(const QString &tag, PeerData *inPeer) {
 | 
						|
		if (MainWidget *m = main()) m->searchMessages(tag + ' ', (inPeer && inPeer->isChannel()) ? inPeer : 0);
 | 
						|
	}
 | 
						|
 | 
						|
	void openPeerByName(const QString &username, bool toProfile, const QString &startToken) {
 | 
						|
		if (MainWidget *m = main()) m->openPeerByName(username, toProfile, startToken);
 | 
						|
	}
 | 
						|
 | 
						|
	void joinGroupByHash(const QString &hash) {
 | 
						|
		if (MainWidget *m = main()) m->joinGroupByHash(hash);
 | 
						|
	}
 | 
						|
 | 
						|
	void stickersBox(const QString &name) {
 | 
						|
		if (MainWidget *m = main()) m->stickersBox(MTP_inputStickerSetShortName(MTP_string(name)));
 | 
						|
	}
 | 
						|
 | 
						|
	void openLocalUrl(const QString &url) {
 | 
						|
		if (MainWidget *m = main()) m->openLocalUrl(url);
 | 
						|
	}
 | 
						|
 | 
						|
	bool forward(const PeerId &peer, ForwardWhatMessages what) {
 | 
						|
		if (MainWidget *m = main()) return m->onForward(peer, what);
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	void removeDialog(History *history) {
 | 
						|
		if (MainWidget *m = main()) m->removeDialog(history);
 | 
						|
	}
 | 
						|
 | 
						|
	void showSettings() {
 | 
						|
		if (Window *win = wnd()) win->showSettings();
 | 
						|
	}
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
namespace Ui {
 | 
						|
 | 
						|
	void showStickerPreview(DocumentData *sticker) {
 | 
						|
		if (MainWidget *m = App::main()) m->ui_showStickerPreview(sticker);
 | 
						|
	}
 | 
						|
 | 
						|
	void hideStickerPreview() {
 | 
						|
		if (MainWidget *m = App::main()) m->ui_hideStickerPreview();
 | 
						|
	}
 | 
						|
 | 
						|
	void showLayer(LayeredWidget *box, ShowLayerOptions options) {
 | 
						|
		if (Window *w = App::wnd()) {
 | 
						|
			w->ui_showLayer(box, options);
 | 
						|
		} else {
 | 
						|
			delete box;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	void hideLayer(bool fast) {
 | 
						|
		if (Window *w = App::wnd()) w->ui_showLayer(0, ShowLayerOptions(CloseOtherLayers) | (fast ? ForceFastShowLayer : AnimatedShowLayer));
 | 
						|
	}
 | 
						|
 | 
						|
	bool isLayerShown() {
 | 
						|
		if (Window *w = App::wnd()) return w->ui_isLayerShown();
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	bool isMediaViewShown() {
 | 
						|
		if (Window *w = App::wnd()) return w->ui_isMediaViewShown();
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	bool isInlineItemBeingChosen() {
 | 
						|
		if (MainWidget *m = App::main()) return m->ui_isInlineItemBeingChosen();
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	void repaintHistoryItem(const HistoryItem *item) {
 | 
						|
		if (!item) return;
 | 
						|
		if (MainWidget *m = App::main()) m->ui_repaintHistoryItem(item);
 | 
						|
	}
 | 
						|
 | 
						|
	void repaintInlineItem(const LayoutInlineItem *layout) {
 | 
						|
		if (!layout) return;
 | 
						|
		if (MainWidget *m = App::main()) m->ui_repaintInlineItem(layout);
 | 
						|
	}
 | 
						|
 | 
						|
	bool isInlineItemVisible(const LayoutInlineItem *layout) {
 | 
						|
		if (MainWidget *m = App::main()) return m->ui_isInlineItemVisible(layout);
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	void showPeerHistory(const PeerId &peer, MsgId msgId, bool back) {
 | 
						|
		if (MainWidget *m = App::main()) m->ui_showPeerHistory(peer, msgId, back);
 | 
						|
	}
 | 
						|
 | 
						|
	void showPeerHistoryAsync(const PeerId &peer, MsgId msgId) {
 | 
						|
		if (MainWidget *m = App::main()) {
 | 
						|
			QMetaObject::invokeMethod(m, "ui_showPeerHistoryAsync", Qt::QueuedConnection, Q_ARG(quint64, peer), Q_ARG(qint32, msgId));
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
namespace Notify {
 | 
						|
 | 
						|
	void userIsBotChanged(UserData *user) {
 | 
						|
		if (MainWidget *m = App::main()) m->notify_userIsBotChanged(user);
 | 
						|
	}
 | 
						|
 | 
						|
	void userIsContactChanged(UserData *user, bool fromThisApp) {
 | 
						|
		if (MainWidget *m = App::main()) m->notify_userIsContactChanged(user, fromThisApp);
 | 
						|
	}
 | 
						|
 | 
						|
	void botCommandsChanged(UserData *user) {
 | 
						|
		if (MainWidget *m = App::main()) m->notify_botCommandsChanged(user);
 | 
						|
	}
 | 
						|
 | 
						|
	void inlineBotRequesting(bool requesting) {
 | 
						|
		if (MainWidget *m = App::main()) m->notify_inlineBotRequesting(requesting);
 | 
						|
	}
 | 
						|
 | 
						|
	void migrateUpdated(PeerData *peer) {
 | 
						|
		if (MainWidget *m = App::main()) m->notify_migrateUpdated(peer);
 | 
						|
	}
 | 
						|
 | 
						|
	void clipStopperHidden(ClipStopperType type) {
 | 
						|
		if (MainWidget *m = App::main()) m->notify_clipStopperHidden(type);
 | 
						|
	}
 | 
						|
 | 
						|
	void historyItemResized(const HistoryItem *item, bool scrollToIt) {
 | 
						|
		if (MainWidget *m = App::main()) m->notify_historyItemResized(item, scrollToIt);
 | 
						|
	}
 | 
						|
 | 
						|
	void historyItemLayoutChanged(const HistoryItem *item) {
 | 
						|
		if (MainWidget *m = App::main()) m->notify_historyItemLayoutChanged(item);
 | 
						|
	}
 | 
						|
 | 
						|
	void automaticLoadSettingsChangedGif() {
 | 
						|
		if (MainWidget *m = App::main()) m->notify_automaticLoadSettingsChangedGif();
 | 
						|
	}
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
struct GlobalDataStruct {
 | 
						|
	QString LangSystemISO;
 | 
						|
	int32 LangSystem = languageDefault;
 | 
						|
 | 
						|
	QByteArray LastCrashDump;
 | 
						|
	ConnectionProxy PreLaunchProxy;
 | 
						|
};
 | 
						|
GlobalDataStruct *GlobalData = 0;
 | 
						|
 | 
						|
namespace Global {
 | 
						|
 | 
						|
	bool CheckBetaVersionDir() {
 | 
						|
		QFile beta(cExeDir() + qsl("TelegramBeta_data/tdata/beta"));
 | 
						|
		if (cBetaVersion()) {
 | 
						|
			cForceWorkingDir(cExeDir() + qsl("TelegramBeta_data/"));
 | 
						|
			if (*BetaPrivateKey) {
 | 
						|
				cSetBetaPrivateKey(QByteArray(BetaPrivateKey));
 | 
						|
			}
 | 
						|
			if (beta.open(QIODevice::WriteOnly)) {
 | 
						|
				QDataStream dataStream(&beta);
 | 
						|
				dataStream.setVersion(QDataStream::Qt_5_3);
 | 
						|
				dataStream << quint64(cRealBetaVersion()) << cBetaPrivateKey();
 | 
						|
			} else {
 | 
						|
				LOG(("Error: could not open '%1' for writing private key!").arg(beta.fileName()));
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
		} else if (beta.exists()) {
 | 
						|
			cForceWorkingDir(cExeDir() + qsl("TelegramBeta_data/"));
 | 
						|
			if (beta.open(QIODevice::ReadOnly)) {
 | 
						|
				QDataStream dataStream(&beta);
 | 
						|
				dataStream.setVersion(QDataStream::Qt_5_3);
 | 
						|
 | 
						|
				quint64 v;
 | 
						|
				QByteArray k;
 | 
						|
				dataStream >> v >> k;
 | 
						|
				if (dataStream.status() == QDataStream::Ok) {
 | 
						|
					cSetBetaVersion(qMax(v, AppVersion * 1000ULL));
 | 
						|
					cSetBetaPrivateKey(k);
 | 
						|
					cSetRealBetaVersion(v);
 | 
						|
				} else {
 | 
						|
					LOG(("Error: '%1' is corrupted, reinstall private beta!").arg(beta.fileName()));
 | 
						|
					return false;
 | 
						|
				}
 | 
						|
			} else {
 | 
						|
				LOG(("Error: could not open '%1' for reading private key!").arg(beta.fileName()));
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
 | 
						|
	void WorkingDirReady() {
 | 
						|
		if (QFile(cWorkingDir() + qsl("tdata/withtestmode")).exists()) {
 | 
						|
			cSetTestMode(true);
 | 
						|
		}
 | 
						|
		if (!cDebug() && QFile(cWorkingDir() + qsl("tdata/withdebug")).exists()) {
 | 
						|
			cSetDebug(true);
 | 
						|
		}
 | 
						|
		if (cBetaVersion()) {
 | 
						|
			cSetDevVersion(false);
 | 
						|
		} else if (!cDevVersion() && QFile(cWorkingDir() + qsl("tdata/devversion")).exists()) {
 | 
						|
			cSetDevVersion(true);
 | 
						|
		} else if (DevVersion) {
 | 
						|
			QFile f(cWorkingDir() + qsl("tdata/devversion"));
 | 
						|
			if (!f.exists() && f.open(QIODevice::WriteOnly)) {
 | 
						|
				f.write("1");
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	void start() {
 | 
						|
		GlobalData = new GlobalDataStruct();
 | 
						|
 | 
						|
		GlobalData->LangSystemISO = psCurrentLanguage();
 | 
						|
		if (GlobalData->LangSystemISO.isEmpty()) GlobalData->LangSystemISO = qstr("en");
 | 
						|
		QByteArray l = LangSystemISO().toLatin1();
 | 
						|
		for (int32 i = 0; i < languageCount; ++i) {
 | 
						|
			if (l.at(0) == LanguageCodes[i][0] && l.at(1) == LanguageCodes[i][1]) {
 | 
						|
				GlobalData->LangSystem = i;
 | 
						|
				break;
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		srand((int32)time(NULL));
 | 
						|
	}
 | 
						|
 | 
						|
	void finish() {
 | 
						|
		delete GlobalData;
 | 
						|
		GlobalData = 0;
 | 
						|
	}
 | 
						|
 | 
						|
#define DefineGlobalReadOnly(Type, Name) const Type &Name() { \
 | 
						|
	t_assert_full(GlobalData != 0, "_data is null in Global::" #Name, __FILE__, __LINE__); \
 | 
						|
	return GlobalData->Name; \
 | 
						|
}
 | 
						|
#define DefineGlobal(Type, Name) DefineGlobalReadOnly(Type, Name) \
 | 
						|
void Set##Name(const Type &Name) { \
 | 
						|
	t_assert_full(GlobalData != 0, "_data is null in Global::Set" #Name, __FILE__, __LINE__); \
 | 
						|
	GlobalData->Name = Name; \
 | 
						|
} \
 | 
						|
Type &Ref##Name() { \
 | 
						|
	t_assert_full(GlobalData != 0, "_data is null in Global::Ref" #Name, __FILE__, __LINE__); \
 | 
						|
	return GlobalData->Name; \
 | 
						|
}
 | 
						|
 | 
						|
	DefineGlobalReadOnly(QString, LangSystemISO);
 | 
						|
	DefineGlobalReadOnly(int32, LangSystem);
 | 
						|
	DefineGlobal(QByteArray, LastCrashDump);
 | 
						|
	DefineGlobal(ConnectionProxy, PreLaunchProxy);
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
struct SandboxDataStruct {
 | 
						|
	uint64 LaunchId = 0;
 | 
						|
};
 | 
						|
SandboxDataStruct *SandboxData = 0;
 | 
						|
 | 
						|
namespace Sandbox {
 | 
						|
 | 
						|
	bool started() {
 | 
						|
		return SandboxData != 0;
 | 
						|
	}
 | 
						|
 | 
						|
	void start() {
 | 
						|
		SandboxData = new SandboxDataStruct();
 | 
						|
 | 
						|
		memset_rand(&SandboxData->LaunchId, sizeof(SandboxData->LaunchId));
 | 
						|
	}
 | 
						|
 | 
						|
	void finish() {
 | 
						|
		delete SandboxData;
 | 
						|
		SandboxData = 0;
 | 
						|
	}
 | 
						|
 | 
						|
#define DefineSandboxReadOnly(Type, Name) const Type &Name() { \
 | 
						|
	t_assert_full(SandboxData != 0, "_data is null in Global::" #Name, __FILE__, __LINE__); \
 | 
						|
	return SandboxData->Name; \
 | 
						|
}
 | 
						|
#define DefineSandbox(Type, Name) DefineSandboxReadOnly(Type, Name) \
 | 
						|
void Set##Name(const Type &Name) { \
 | 
						|
	t_assert_full(SandboxData != 0, "_data is null in Global::Set" #Name, __FILE__, __LINE__); \
 | 
						|
	SandboxData->Name = Name; \
 | 
						|
} \
 | 
						|
Type &Ref##Name() { \
 | 
						|
	t_assert_full(SandboxData != 0, "_data is null in Global::Ref" #Name, __FILE__, __LINE__); \
 | 
						|
	return SandboxData->Name; \
 | 
						|
}
 | 
						|
 | 
						|
	DefineSandboxReadOnly(uint64, LaunchId);
 | 
						|
 | 
						|
};
 |