/* 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/")); QDir().mkpath(cWorkingDir() + qstr("tdata")); 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(("FATAL: 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(("FATAL: '%1' is corrupted, reinstall private beta!").arg(beta.fileName())); return false; } } else { LOG(("FATAL: 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 DefineSandboxRef(Type, Name) DefineSandboxReadOnly(Type, Name) \ Type &Ref##Name() { \ t_assert_full(SandboxData != 0, "_data is null in Global::Ref" #Name, __FILE__, __LINE__); \ return SandboxData->Name; \ } #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; \ } DefineSandboxReadOnly(uint64, LaunchId); };