2015-12-07 13:05:00 +00:00
|
|
|
/*
|
|
|
|
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"
|
|
|
|
|
2015-12-24 20:29:33 +00:00
|
|
|
#include "layerwidget.h"
|
2016-01-11 15:43:29 +00:00
|
|
|
#include "lang.h"
|
2015-12-24 20:29:33 +00:00
|
|
|
|
2015-12-07 13:05:00 +00:00
|
|
|
namespace App {
|
|
|
|
|
|
|
|
void sendBotCommand(const QString &cmd, MsgId replyTo) {
|
|
|
|
if (MainWidget *m = main()) m->sendBotCommand(cmd, replyTo);
|
|
|
|
}
|
|
|
|
|
2015-12-31 18:34:56 +00:00
|
|
|
bool insertBotCommand(const QString &cmd, bool specialGif) {
|
|
|
|
if (MainWidget *m = main()) return m->insertBotCommand(cmd, specialGif);
|
|
|
|
return false;
|
2015-12-07 13:05:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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();
|
|
|
|
}
|
|
|
|
|
2015-12-07 18:09:05 +00:00
|
|
|
void showLayer(LayeredWidget *box, ShowLayerOptions options) {
|
2015-12-16 15:04:02 +00:00
|
|
|
if (Window *w = App::wnd()) {
|
|
|
|
w->ui_showLayer(box, options);
|
|
|
|
} else {
|
|
|
|
delete box;
|
|
|
|
}
|
2015-12-07 18:09:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2015-12-25 13:09:14 +00:00
|
|
|
bool isMediaViewShown() {
|
|
|
|
if (Window *w = App::wnd()) return w->ui_isMediaViewShown();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-12-30 19:09:20 +00:00
|
|
|
bool isInlineItemBeingChosen() {
|
|
|
|
if (MainWidget *m = App::main()) return m->ui_isInlineItemBeingChosen();
|
2015-12-27 21:37:48 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void repaintHistoryItem(const HistoryItem *item) {
|
|
|
|
if (!item) return;
|
|
|
|
if (MainWidget *m = App::main()) m->ui_repaintHistoryItem(item);
|
|
|
|
}
|
|
|
|
|
2015-12-30 19:09:20 +00:00
|
|
|
void repaintInlineItem(const LayoutInlineItem *layout) {
|
2015-12-27 21:37:48 +00:00
|
|
|
if (!layout) return;
|
2015-12-30 19:09:20 +00:00
|
|
|
if (MainWidget *m = App::main()) m->ui_repaintInlineItem(layout);
|
2015-12-22 08:01:02 +00:00
|
|
|
}
|
|
|
|
|
2015-12-30 19:09:20 +00:00
|
|
|
bool isInlineItemVisible(const LayoutInlineItem *layout) {
|
|
|
|
if (MainWidget *m = App::main()) return m->ui_isInlineItemVisible(layout);
|
2015-12-27 21:37:48 +00:00
|
|
|
return false;
|
2015-12-13 11:17:15 +00:00
|
|
|
}
|
|
|
|
|
2015-12-12 22:29:33 +00:00
|
|
|
void showPeerHistory(const PeerId &peer, MsgId msgId, bool back) {
|
2015-12-13 11:36:08 +00:00
|
|
|
if (MainWidget *m = App::main()) m->ui_showPeerHistory(peer, msgId, back);
|
2015-12-12 22:29:33 +00:00
|
|
|
}
|
|
|
|
|
2015-12-13 11:17:15 +00:00
|
|
|
void showPeerHistoryAsync(const PeerId &peer, MsgId msgId) {
|
|
|
|
if (MainWidget *m = App::main()) {
|
2016-01-01 14:48:32 +00:00
|
|
|
QMetaObject::invokeMethod(m, "ui_showPeerHistoryAsync", Qt::QueuedConnection, Q_ARG(quint64, peer), Q_ARG(qint32, msgId));
|
2015-12-13 11:17:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-07 13:05:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace Notify {
|
|
|
|
|
|
|
|
void userIsBotChanged(UserData *user) {
|
2015-12-11 18:11:38 +00:00
|
|
|
if (MainWidget *m = App::main()) m->notify_userIsBotChanged(user);
|
2015-12-07 13:05:00 +00:00
|
|
|
}
|
|
|
|
|
2015-12-13 11:17:15 +00:00
|
|
|
void userIsContactChanged(UserData *user, bool fromThisApp) {
|
|
|
|
if (MainWidget *m = App::main()) m->notify_userIsContactChanged(user, fromThisApp);
|
|
|
|
}
|
|
|
|
|
2015-12-07 13:05:00 +00:00
|
|
|
void botCommandsChanged(UserData *user) {
|
2015-12-11 18:11:38 +00:00
|
|
|
if (MainWidget *m = App::main()) m->notify_botCommandsChanged(user);
|
2015-12-07 13:05:00 +00:00
|
|
|
}
|
|
|
|
|
2016-01-01 09:58:05 +00:00
|
|
|
void inlineBotRequesting(bool requesting) {
|
|
|
|
if (MainWidget *m = App::main()) m->notify_inlineBotRequesting(requesting);
|
|
|
|
}
|
|
|
|
|
2015-12-07 13:05:00 +00:00
|
|
|
void migrateUpdated(PeerData *peer) {
|
2015-12-11 18:11:38 +00:00
|
|
|
if (MainWidget *m = App::main()) m->notify_migrateUpdated(peer);
|
|
|
|
}
|
|
|
|
|
2015-12-27 21:37:48 +00:00
|
|
|
void clipStopperHidden(ClipStopperType type) {
|
|
|
|
if (MainWidget *m = App::main()) m->notify_clipStopperHidden(type);
|
2015-12-25 13:09:14 +00:00
|
|
|
}
|
|
|
|
|
2015-12-22 08:01:02 +00:00
|
|
|
void historyItemResized(const HistoryItem *item, bool scrollToIt) {
|
|
|
|
if (MainWidget *m = App::main()) m->notify_historyItemResized(item, scrollToIt);
|
|
|
|
}
|
|
|
|
|
2015-12-12 22:29:33 +00:00
|
|
|
void historyItemLayoutChanged(const HistoryItem *item) {
|
|
|
|
if (MainWidget *m = App::main()) m->notify_historyItemLayoutChanged(item);
|
|
|
|
}
|
|
|
|
|
2015-12-31 15:27:21 +00:00
|
|
|
void automaticLoadSettingsChangedGif() {
|
|
|
|
if (MainWidget *m = App::main()) m->notify_automaticLoadSettingsChangedGif();
|
|
|
|
}
|
|
|
|
|
2015-12-07 13:05:00 +00:00
|
|
|
}
|
2016-01-09 11:24:16 +00:00
|
|
|
|
2016-01-11 15:43:29 +00:00
|
|
|
struct GlobalDataStruct {
|
|
|
|
QString LangSystemISO;
|
|
|
|
int32 LangSystem = languageDefault;
|
2016-01-21 06:58:58 +00:00
|
|
|
|
|
|
|
QByteArray LastCrashDump;
|
2016-01-30 16:31:10 +00:00
|
|
|
ConnectionProxy PreLaunchProxy;
|
2016-01-11 15:43:29 +00:00
|
|
|
};
|
|
|
|
GlobalDataStruct *GlobalData = 0;
|
|
|
|
|
2016-01-09 11:24:16 +00:00
|
|
|
namespace Global {
|
|
|
|
|
2016-01-11 15:43:29 +00:00
|
|
|
bool CheckBetaVersionDir() {
|
|
|
|
QFile beta(cExeDir() + qsl("TelegramBeta_data/tdata/beta"));
|
|
|
|
if (cBetaVersion()) {
|
|
|
|
cForceWorkingDir(cExeDir() + qsl("TelegramBeta_data/"));
|
2016-02-04 22:16:25 +00:00
|
|
|
QDir().mkpath(cWorkingDir() + qstr("tdata"));
|
2016-01-11 15:43:29 +00:00
|
|
|
if (*BetaPrivateKey) {
|
|
|
|
cSetBetaPrivateKey(QByteArray(BetaPrivateKey));
|
|
|
|
}
|
|
|
|
if (beta.open(QIODevice::WriteOnly)) {
|
|
|
|
QDataStream dataStream(&beta);
|
|
|
|
dataStream.setVersion(QDataStream::Qt_5_3);
|
|
|
|
dataStream << quint64(cRealBetaVersion()) << cBetaPrivateKey();
|
|
|
|
} else {
|
2016-02-04 12:44:39 +00:00
|
|
|
LOG(("FATAL: Could not open '%1' for writing private key!").arg(beta.fileName()));
|
2016-01-11 15:43:29 +00:00
|
|
|
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 {
|
2016-02-04 12:44:39 +00:00
|
|
|
LOG(("FATAL: '%1' is corrupted, reinstall private beta!").arg(beta.fileName()));
|
2016-01-11 15:43:29 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
2016-02-04 12:44:39 +00:00
|
|
|
LOG(("FATAL: could not open '%1' for reading private key!").arg(beta.fileName()));
|
2016-01-11 15:43:29 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2016-01-09 11:24:16 +00:00
|
|
|
|
2016-01-11 15:43:29 +00:00
|
|
|
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");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-01-09 11:24:16 +00:00
|
|
|
|
2016-01-11 15:43:29 +00:00
|
|
|
void start() {
|
|
|
|
GlobalData = new GlobalDataStruct();
|
2016-01-09 12:51:42 +00:00
|
|
|
|
2016-01-11 15:43:29 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
2016-01-30 16:31:10 +00:00
|
|
|
|
|
|
|
srand((int32)time(NULL));
|
2016-01-09 11:24:16 +00:00
|
|
|
}
|
|
|
|
|
2016-01-11 15:43:29 +00:00
|
|
|
void finish() {
|
|
|
|
delete GlobalData;
|
|
|
|
GlobalData = 0;
|
2016-01-09 11:24:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#define DefineGlobalReadOnly(Type, Name) const Type &Name() { \
|
2016-01-11 15:43:29 +00:00
|
|
|
t_assert_full(GlobalData != 0, "_data is null in Global::" #Name, __FILE__, __LINE__); \
|
|
|
|
return GlobalData->Name; \
|
2016-01-09 11:24:16 +00:00
|
|
|
}
|
|
|
|
#define DefineGlobal(Type, Name) DefineGlobalReadOnly(Type, Name) \
|
|
|
|
void Set##Name(const Type &Name) { \
|
2016-01-11 15:43:29 +00:00
|
|
|
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);
|
2016-01-21 06:58:58 +00:00
|
|
|
DefineGlobal(QByteArray, LastCrashDump);
|
2016-01-30 16:31:10 +00:00
|
|
|
DefineGlobal(ConnectionProxy, PreLaunchProxy);
|
2016-01-11 15:43:29 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
struct SandboxDataStruct {
|
|
|
|
uint64 LaunchId = 0;
|
|
|
|
};
|
|
|
|
SandboxDataStruct *SandboxData = 0;
|
|
|
|
|
|
|
|
namespace Sandbox {
|
|
|
|
|
2016-01-30 16:31:10 +00:00
|
|
|
bool started() {
|
|
|
|
return SandboxData != 0;
|
|
|
|
}
|
|
|
|
|
2016-01-11 15:43:29 +00:00
|
|
|
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; \
|
|
|
|
}
|
2016-02-07 18:58:42 +00:00
|
|
|
#define DefineSandboxRef(Type, Name) DefineSandboxReadOnly(Type, Name) \
|
2016-01-09 11:24:16 +00:00
|
|
|
Type &Ref##Name() { \
|
2016-01-11 15:43:29 +00:00
|
|
|
t_assert_full(SandboxData != 0, "_data is null in Global::Ref" #Name, __FILE__, __LINE__); \
|
|
|
|
return SandboxData->Name; \
|
2016-01-09 11:24:16 +00:00
|
|
|
}
|
2016-02-07 18:58:42 +00:00
|
|
|
#define DefineSandbox(Type, Name) DefineSandboxRef(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; \
|
|
|
|
}
|
2016-01-09 11:24:16 +00:00
|
|
|
|
2016-01-11 15:43:29 +00:00
|
|
|
DefineSandboxReadOnly(uint64, LaunchId);
|
2016-01-09 11:24:16 +00:00
|
|
|
|
|
|
|
};
|