/* This file is part of Telegram Desktop, an unofficial desktop 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. Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE Copyright (c) 2014 John Preston, https://tdesktop.com */ #include "stdafx.h" #include "style.h" #include "lang.h" #include "window.h" #include "application.h" #include "pspecific.h" #include "title.h" #include "intro/intro.h" #include "mainwidget.h" #include "layerwidget.h" #include "settingswidget.h" ConnectingWidget::ConnectingWidget(QWidget *parent, const QString &text, const QString &reconnect) : QWidget(parent), _shadow(st::boxShadow), _reconnect(this, QString()) { set(text, reconnect); connect(&_reconnect, SIGNAL(clicked()), this, SLOT(onReconnect())); } void ConnectingWidget::set(const QString &text, const QString &reconnect) { _text = text; _textWidth = st::linkFont->m.width(_text) + st::linkFont->spacew; int32 _reconnectWidth = 0; if (reconnect.isEmpty()) { _reconnect.hide(); } else { _reconnect.setText(reconnect); _reconnect.show(); _reconnect.move(st::connectingPadding.left() + _textWidth, st::boxShadow.pxHeight() + st::connectingPadding.top()); _reconnectWidth = _reconnect.width(); } resize(st::connectingPadding.left() + _textWidth + _reconnectWidth + st::connectingPadding.right() + st::boxShadow.pxWidth(), st::boxShadow.pxHeight() + st::connectingPadding.top() + st::linkFont->height + st::connectingPadding.bottom()); update(); } void ConnectingWidget::paintEvent(QPaintEvent *e) { QPainter p(this); _shadow.paint(p, QRect(0, st::boxShadow.pxHeight(), width() - st::boxShadow.pxWidth(), height() - st::boxShadow.pxHeight()), QPoint(0, 0), BoxShadow::Top | BoxShadow::Right); p.fillRect(0, st::boxShadow.pxHeight(), width() - st::boxShadow.pxWidth(), height() - st::boxShadow.pxHeight(), st::connectingBG->b); p.setFont(st::linkFont->f); p.setPen(st::connectingColor->p); p.drawText(st::connectingPadding.left(), st::boxShadow.pxHeight() + st::connectingPadding.top() + st::linkFont->ascent, _text); } void ConnectingWidget::onReconnect() { MTP::restart(); } TempDirDeleter::TempDirDeleter(QThread *thread) { moveToThread(thread); connect(thread, SIGNAL(started()), this, SLOT(onStart())); } void TempDirDeleter::onStart() { if (QDir(cTempDir()).removeRecursively()) { emit succeed(); } else { emit failed(); } } NotifyWindow::NotifyWindow(HistoryItem *msg, int32 x, int32 y) : history(msg->history()), item(msg) #ifdef Q_OS_WIN , started(GetTickCount()) #endif , close(this, st::notifyClose) , alphaDuration(st::notifyFastAnim) , posDuration(st::notifyFastAnim) , hiding(false) , _index(0) , aOpacity(0) , aOpacityFunc(st::notifyFastAnimFunc) , aY(y + st::notifyHeight + st::notifyDeltaY) { updateNotifyDisplay(); hideTimer.setSingleShot(true); connect(&hideTimer, SIGNAL(timeout()), this, SLOT(hideByTimer())); inputTimer.setSingleShot(true); connect(&inputTimer, SIGNAL(timeout()), this, SLOT(checkLastInput())); connect(&close, SIGNAL(clicked()), this, SLOT(unlinkHistory())); close.setAcceptBoth(true); close.move(st::notifyWidth - st::notifyClose.width - st::notifyClosePos.x(), st::notifyClosePos.y()); close.show(); aY.start(y); setGeometry(x, aY.current(), st::notifyWidth, st::notifyHeight); aOpacity.start(1); setWindowFlags(Qt::Tool | Qt::WindowStaysOnTopHint | Qt::FramelessWindowHint | Qt::X11BypassWindowManagerHint); setAttribute(Qt::WA_MacAlwaysShowToolWindow); show(); setWindowOpacity(aOpacity.current()); alphaDuration = posDuration = st::notifyFastAnim; anim::start(this); checkLastInput(); } void NotifyWindow::checkLastInput() { #ifdef Q_OS_WIN LASTINPUTINFO lii; lii.cbSize = sizeof(LASTINPUTINFO); BOOL res = GetLastInputInfo(&lii); if (!res || lii.dwTime >= started) { hideTimer.start(st::notifyWaitLongHide); } else { inputTimer.start(300); } #else // TODO if (true) { hideTimer.start(st::notifyWaitLongHide); } else { inputTimer.start(300); } #endif } void NotifyWindow::moveTo(int32 x, int32 y, int32 index) { if (index >= 0) { _index = index; } move(x, aY.current()); aY.start(y); aOpacity.restart(); posDuration = st::notifyFastAnim; anim::start(this); } void NotifyWindow::updateNotifyDisplay() { if (!item) return; int32 w = st::notifyWidth, h = st::notifyHeight; QImage img(w * cIntRetinaFactor(), h * cIntRetinaFactor(), QImage::Format_ARGB32_Premultiplied); if (cRetina()) img.setDevicePixelRatio(cRetinaFactor()); img.fill(st::notifyBG->c); { QPainter p(&img); p.fillRect(0, 0, w - st::notifyBorderWidth, st::notifyBorderWidth, st::notifyBorder->b); p.fillRect(w - st::notifyBorderWidth, 0, st::notifyBorderWidth, h - st::notifyBorderWidth, st::notifyBorder->b); p.fillRect(st::notifyBorderWidth, h - st::notifyBorderWidth, w - st::notifyBorderWidth, st::notifyBorderWidth, st::notifyBorder->b); p.fillRect(0, st::notifyBorderWidth, st::notifyBorderWidth, h - st::notifyBorderWidth, st::notifyBorder->b); if (cNotifyView() <= dbinvShowName) { if (history->peer->photo->loaded()) { p.drawPixmap(st::notifyPhotoPos.x(), st::notifyPhotoPos.y(), history->peer->photo->pix(st::notifyPhotoSize)); } else { MTP::clearLoaderPriorities(); peerPhoto = history->peer->photo; peerPhoto->load(true, true); } } else { static QPixmap icon = QPixmap::fromImage(App::wnd()->iconLarge().scaled(st::notifyPhotoSize, st::notifyPhotoSize, Qt::IgnoreAspectRatio, Qt::SmoothTransformation)); p.drawPixmap(st::notifyPhotoPos.x(), st::notifyPhotoPos.y(), icon); } int32 itemWidth = w - st::notifyPhotoPos.x() - st::notifyPhotoSize - st::notifyTextLeft - st::notifyClosePos.x() - st::notifyClose.width; QRect rectForName(st::notifyPhotoPos.x() + st::notifyPhotoSize + st::notifyTextLeft, st::notifyTextTop, itemWidth, st::msgNameFont->height); if (cNotifyView() <= dbinvShowName) { if (history->peer->chat) { p.drawPixmap(QPoint(rectForName.left() + st::dlgChatImgLeft, rectForName.top() + st::dlgChatImgTop), App::sprite(), st::dlgChatImg); rectForName.setLeft(rectForName.left() + st::dlgChatImgSkip); } } QDateTime now(QDateTime::currentDateTime()), lastTime(item->date); QDate nowDate(now.date()), lastDate(lastTime.date()); QString dt = lastTime.toString(qsl("hh:mm")); int32 dtWidth = st::dlgHistFont->m.width(dt); rectForName.setWidth(rectForName.width() - dtWidth - st::dlgDateSkip); p.setFont(st::dlgDateFont->f); p.setPen(st::dlgDateColor->p); p.drawText(rectForName.left() + rectForName.width() + st::dlgDateSkip, rectForName.top() + st::dlgHistFont->ascent, dt); if (cNotifyView() <= dbinvShowPreview) { const HistoryItem *textCachedFor = 0; Text itemTextCache(itemWidth); bool active = false; item->drawInDialog(p, QRect(st::notifyPhotoPos.x() + st::notifyPhotoSize + st::notifyTextLeft, st::notifyItemTop + st::msgNameFont->height, itemWidth, 2 * st::dlgFont->height), active, textCachedFor, itemTextCache); } else { static QString notifyText = st::dlgHistFont->m.elidedText(lang(lng_notification_preview), Qt::ElideRight, itemWidth); p.setPen(st::dlgSystemColor->p); p.drawText(st::notifyPhotoPos.x() + st::notifyPhotoSize + st::notifyTextLeft, st::notifyItemTop + st::msgNameFont->height + st::dlgHistFont->ascent, notifyText); } p.setPen(st::dlgNameColor->p); if (cNotifyView() <= dbinvShowName) { history->nameText.drawElided(p, rectForName.left(), rectForName.top(), rectForName.width()); } else { p.setFont(st::msgNameFont->f); static QString notifyTitle = st::msgNameFont->m.elidedText(lang(lng_notification_title), Qt::ElideRight, rectForName.width()); p.drawText(rectForName.left(), rectForName.top() + st::msgNameFont->ascent, notifyTitle); } } pm = QPixmap::fromImage(img); update(); } void NotifyWindow::updatePeerPhoto() { if (!peerPhoto->isNull() && peerPhoto->loaded()) { QImage img(pm.toImage()); { QPainter p(&img); p.drawPixmap(st::notifyPhotoPos.x(), st::notifyPhotoPos.y(), peerPhoto->pix(st::notifyPhotoSize)); } peerPhoto = ImagePtr(); pm = QPixmap::fromImage(img); update(); } } void NotifyWindow::itemRemoved(HistoryItem *del) { if (item == del) { unlinkHistory(); } } void NotifyWindow::unlinkHistory(History *hist) { if (!hist || hist == history) { animHide(st::notifyFastAnim, st::notifyFastAnimFunc); history = 0; item = 0; App::wnd()->notifyShowNext(); } } void NotifyWindow::enterEvent(QEvent *e) { if (!history) return; if (App::wnd()) App::wnd()->notifyStopHiding(); } void NotifyWindow::leaveEvent(QEvent *e) { if (!history) return; App::wnd()->notifyStartHiding(); } void NotifyWindow::startHiding() { hideTimer.start(st::notifyWaitShortHide); } void NotifyWindow::mousePressEvent(QMouseEvent *e) { if (!history) return; if (e->button() == Qt::RightButton) { unlinkHistory(); } else if (history) { App::wnd()->showFromTray(); App::wnd()->hideSettings(); App::main()->showPeer(history->peer->id, 0, false, true); e->ignore(); } } void NotifyWindow::paintEvent(QPaintEvent *e) { QPainter p(this); p.drawPixmap(0, 0, pm); } void NotifyWindow::animHide(float64 duration, anim::transition func) { if (!history) return; alphaDuration = duration; aOpacityFunc = func; aOpacity.start(0); aY.restart(); hiding = true; anim::start(this); } void NotifyWindow::stopHiding() { if (!history) return; alphaDuration = st::notifyFastAnim; aOpacityFunc = st::notifyFastAnimFunc; aOpacity.start(1); aY.restart(); hiding = false; hideTimer.stop(); anim::start(this); } void NotifyWindow::hideByTimer() { if (!history) return; animHide(st::notifySlowHide, st::notifySlowHideFunc); } bool NotifyWindow::animStep(float64 ms) { float64 dtAlpha = ms / alphaDuration, dtPos = ms / posDuration; if (dtAlpha >= 1) { aOpacity.finish(); if (hiding) { deleteLater(); } } else { aOpacity.update(dtAlpha, aOpacityFunc); } setWindowOpacity(aOpacity.current()); if (dtPos >= 1) { aY.finish(); } else { aY.update(dtPos, anim::linear); } move(x(), aY.current()); update(); return (dtAlpha < 1 || (!hiding && dtPos < 1)); } NotifyWindow::~NotifyWindow() { if (App::wnd()) App::wnd()->notifyShowNext(this); } Window::Window(QWidget *parent) : PsMainWindow(parent), intro(0), main(0), settings(0), layer(0), layerBG(0), _topWidget(0), _connecting(0), _tempDeleter(0), _tempDeleterThread(0), myIcon(QPixmap::fromImage(icon256)), dragging(false), _inactivePress(false) { if (objectName().isEmpty()) setObjectName(qsl("MainWindow")); resize(st::wndDefWidth, st::wndDefHeight); setWindowOpacity(1); setLocale(QLocale(QLocale::English, QLocale::UnitedStates)); centralwidget = new QWidget(this); centralwidget->setObjectName(qsl("centralwidget")); setCentralWidget(centralwidget); QMetaObject::connectSlotsByName(this); _inactiveTimer.setSingleShot(true); connect(&_inactiveTimer, SIGNAL(timeout()), this, SLOT(onInactiveTimer())); connect(¬ifyWaitTimer, SIGNAL(timeout()), this, SLOT(notifyFire())); notifyWaitTimer.setSingleShot(true); } void Window::inactivePress(bool inactive) { _inactivePress = inactive; if (_inactivePress) { _inactiveTimer.start(200); } else { _inactiveTimer.stop(); } } bool Window::inactivePress() const { return _inactivePress; } void Window::onInactiveTimer() { inactivePress(false); } void Window::init() { psInitFrameless(); setWindowIcon(myIcon); App::app()->installEventFilter(this); connect(windowHandle(), SIGNAL(activeChanged()), this, SLOT(checkHistoryActivation())); QPalette p(palette()); p.setColor(QPalette::Window, st::wndBG->c); setPalette(p); title = new TitleWidget(this); psInitSize(); psUpdateWorkmode(); } void Window::clearWidgets() { layerHidden(); if (settings) { anim::stop(settings); settings->hide(); settings->deleteLater(); settings = 0; } if (main) { anim::stop(main); main->hide(); main->deleteLater(); main = 0; } if (intro) { anim::stop(intro); intro->hide(); intro->deleteLater(); intro = 0; } } void Window::setupIntro(bool anim) { if (intro && (intro->animating() || intro->isVisible()) && !main) return; QPixmap bg = myGrab(this, QRect(0, st::titleHeight, width(), height() - st::titleHeight)); clearWidgets(); intro = new IntroWidget(this); intro->move(0, st::titleHeight); if (anim) { intro->animShow(bg); } fixOrder(); updateTitleStatus(); } void Window::getNotifySetting(const MTPInputNotifyPeer &peer, uint32 msWait) { MTP::send(MTPaccount_GetNotifySettings(peer), main->rpcDone(&MainWidget::gotNotifySetting, peer), main->rpcFail(&MainWidget::failNotifySetting, peer), 0, msWait); } void Window::setupMain(bool anim) { QPixmap bg = myGrab(this, QRect(0, st::titleHeight, width(), height() - st::titleHeight)); clearWidgets(); main = new MainWidget(this); main->move(0, st::titleHeight); if (anim) { main->animShow(bg); } else { MTP::send(MTPusers_GetUsers(MTP_vector(QVector(1, MTP_inputUserSelf()))), main->rpcDone(&MainWidget::startFull)); main->activate(); } fixOrder(); updateTitleStatus(); } void Window::showSettings() { App::wnd()->hideLayer(); if (settings) { return hideSettings(); } QPixmap bg = myGrab(this, QRect(0, st::titleHeight, width(), height() - st::titleHeight)); if (intro) { anim::stop(intro); intro->hide(); } else if (main) { anim::stop(main); main->hide(); } settings = new Settings(this); settings->animShow(bg); fixOrder(); } void Window::hideSettings(bool fast) { if (!settings) return; if (fast) { anim::stop(settings); settings->hide(); settings->deleteLater(); settings = 0; if (intro) { intro->show(); } else { main->show(); } } else { QPixmap bg = myGrab(this, QRect(0, st::titleHeight, width(), height() - st::titleHeight)); anim::stop(settings); settings->hide(); settings->deleteLater(); settings = 0; if (intro) { intro->animShow(bg, true); } else { main->animShow(bg, true); } } fixOrder(); } void Window::startMain(const MTPUser &user) { if (main) main->start(user); title->resizeEvent(0); } void Window::mtpStateChanged(int32 dc, int32 state) { if (dc == MTP::maindc()) { updateTitleStatus(); if (settings) settings->updateConnectionType(); } } void Window::updateTitleStatus() { int32 state = MTP::dcstate(); if (state == MTProtoConnection::Connecting || state == MTProtoConnection::Disconnected || (state < 0 && state > -600)) { if (main || getms() > 5000 || _connecting) { showConnecting(lang(lng_connecting)); } } else if (state < 0) { showConnecting(lang(lng_reconnecting).arg((-state) / 1000), lang(lng_reconnecting_try_now)); QTimer::singleShot((-state) % 1000, this, SLOT(updateTitleStatus())); } else { hideConnecting(); } } IntroWidget *Window::introWidget() { return intro; } MainWidget *Window::mainWidget() { return main; } Settings *Window::settingsWidget() { return settings; } void Window::showPhoto(const PhotoLink *lnk, HistoryItem *item) { return showPhoto(lnk->photo(), item); } void Window::showPhoto(PhotoData *photo, HistoryItem *item) { layerHidden(); layer = new LayerWidget(this, photo, item); } PhotoData *Window::photoShown() { return layer ? layer->photoShown() : 0; } /* void Window::showVideo(const VideoOpenLink *lnk, HistoryItem *item) { layerHidden(); VideoData *video = App::video(lnk->video()); layer = new LayerWidget(this, video, item); } /**/ void Window::showLayer(LayeredWidget *w) { layerHidden(); layerBG = new BackgroundWidget(this, w); } void Window::showConnecting(const QString &text, const QString &reconnect) { if (_connecting) { _connecting->set(text, reconnect); } else { _connecting = new ConnectingWidget(this, text, reconnect); resizeEvent(0); } } void Window::hideConnecting() { if (_connecting) { _connecting->deleteLater(); _connecting = 0; } } void Window::replaceLayer(LayeredWidget *w) { if (layer) layer->deleteLater(); layer = 0; if (layerBG) { layerBG->replaceInner(w); } else { layerBG = new BackgroundWidget(this, w); } } void Window::hideLayer() { if (layerBG) { layerBG->onClose(); } if (layer) { layer->startHide(); } } bool Window::layerShown() { return !!layer || !!layerBG || !!_topWidget; } bool Window::historyIsActive(int state) const { return psIsActive(state) && main && main->historyIsActive() && (!settings || !settings->isVisible()); } void Window::checkHistoryActivation(int state) { if (main && MTP::authedId() && historyIsActive(state)) { main->historyWasRead(); } } void Window::layerHidden() { if (layer) layer->deleteLater(); layer = 0; if (layerBG) layerBG->deleteLater(); layerBG = 0; if (main) main->setInnerFocus(); } QRect Window::clientRect() const { return QRect(0, st::titleHeight, width(), height() - st::titleHeight); } QRect Window::photoRect() const { if (settings) { return settings->geometry(); } else if (main) { QRect r(main->historyRect()); r.moveLeft(r.left() + main->x()); r.moveTop(r.top() + main->y()); return r; } return QRect(0, 0, 0, 0); } void Window::wStartDrag(QMouseEvent *e) { dragStart = e->globalPos() - frameGeometry().topLeft(); dragging = true; } void Window::paintEvent(QPaintEvent *e) { } HitTestType Window::hitTest(const QPoint &p) const { int x(p.x()), y(p.y()), w(width()), h(height()); const int32 raw = psResizeRowWidth(); if (!windowState().testFlag(Qt::WindowMaximized)) { if (y < raw) { if (x < raw) { return HitTestTopLeft; } else if (x > w - raw - 1) { return HitTestTopRight; } return HitTestTop; } else if (y > h - raw - 1) { if (x < raw) { return HitTestBottomLeft; } else if (x > w - raw - 1) { return HitTestBottomRight; } return HitTestBottom; } else if (x < raw) { return HitTestLeft; } else if (x > w - raw - 1) { return HitTestRight; } } HitTestType titleTest = title->hitTest(p - title->geometry().topLeft()); if (titleTest && (!layer || titleTest != HitTestCaption)) { return titleTest; } else if (x >= 0 && y >= 0 && x < w && y < h) { return HitTestClient; } return HitTestNone; } bool Window::getPhotoCoords(PhotoData *photo, int32 &x, int32 &y, int32 &w) const { if (main && main->getPhotoCoords(photo, x, y, w)) { x += main->x(); y += main->y(); return true; } else if (settings && settings->getPhotoCoords(photo, x, y, w)) { x += main->x(); y += main->y(); return true; } return false; } bool Window::getVideoCoords(VideoData *video, int32 &x, int32 &y, int32 &w) const { if (main && main->getVideoCoords(video, x, y, w)) { x += main->x(); y += main->y(); return true; } return false; } QRect Window::iconRect() const { return QRect(st::titleIconPos + title->geometry().topLeft(), st::titleIconRect.pxSize()); } bool Window::eventFilter(QObject *obj, QEvent *evt) { if (obj == App::app() && (evt->type() == QEvent::ApplicationActivate)) { QTimer::singleShot(1, this, SLOT(checkHistoryActivation())); } else if (obj == this && evt->type() == QEvent::WindowStateChange) { Qt::WindowState state = (windowState() & Qt::WindowMinimized) ? Qt::WindowMinimized : ((windowState() & Qt::WindowMaximized) ? Qt::WindowMaximized : ((windowState() & Qt::WindowFullScreen) ? Qt::WindowFullScreen : Qt::WindowNoState)); psStateChanged(state); if (App::main()) { App::main()->mainStateChanged(state); } } else if (obj == this && (evt->type() == QEvent::Move || evt->type() == QEvent::Resize)) { psUpdatedPosition(); } return PsMainWindow::eventFilter(obj, evt); } void Window::mouseMoveEvent(QMouseEvent *e) { if (e->buttons() & Qt::LeftButton) { if (dragging) { if (windowState().testFlag(Qt::WindowMaximized)) { setWindowState(windowState() & ~Qt::WindowMaximized); dragStart = e->globalPos() - frameGeometry().topLeft(); } else { move(e->globalPos() - dragStart); } } } else if (dragging) { dragging = false; } } void Window::mouseReleaseEvent(QMouseEvent *e) { dragging = false; } bool Window::minimizeToTray() { if (App::quiting() || !trayIcon) return false; hide(); if (!cSeenTrayTooltip()) { trayIcon->showMessage(QString::fromStdWString(AppName), lang(lng_tray_icon_text), QSystemTrayIcon::Information, 10000); cSetSeenTrayTooltip(true); App::writeConfig(); } if (App::main()) App::main()->setOnline(windowState()); return true; } void Window::setupTrayIcon() { if (!trayIcon) { if (trayIconMenu) trayIconMenu->deleteLater(); trayIconMenu = new QMenu(this); trayIconMenu->setFont(QFont("Tahoma")); QAction *a; a = trayIconMenu->addAction(lang(lng_open_from_tray), this, SLOT(showFromTray())); a->setEnabled(true); a = trayIconMenu->addAction(lang(lng_quit_from_tray), this, SLOT(quitFromTray())); a->setEnabled(true); if (trayIcon) trayIcon->deleteLater(); trayIcon = new QSystemTrayIcon(this); trayIcon->setIcon(this->windowIcon()); trayIcon->setContextMenu(trayIconMenu); trayIcon->setToolTip(QString::fromStdWString(AppName)); connect(trayIcon, SIGNAL(activated(QSystemTrayIcon::ActivationReason)), this, SLOT(toggleTray(QSystemTrayIcon::ActivationReason))); connect(trayIcon, SIGNAL(messageClicked()), this, SLOT(showFromTray())); } psUpdateCounter(); trayIcon->show(); } void Window::quitFromTray() { App::quit(); } void Window::activate() { bool wasHidden = !isVisible(); setWindowState(windowState() & ~Qt::WindowMinimized); setVisible(true); activateWindow(); if (wasHidden) { if (main) { main->windowShown(); } } } void Window::noIntro(IntroWidget *was) { if (was == intro) { intro = 0; } } void Window::noSettings(Settings *was) { if (was == settings) { settings = 0; } checkHistoryActivation(); } void Window::noMain(MainWidget *was) { if (was == main) { main = 0; } } void Window::noLayer(LayerWidget *was) { if (was == layer) { layer = 0; } fixOrder(); } void Window::noBox(BackgroundWidget *was) { if (was == layerBG) { layerBG = 0; } } void Window::fixOrder() { title->raise(); if (layer) layer->raise(); if (layerBG) layerBG->raise(); if (_topWidget) _topWidget->raise(); if (_connecting) _connecting->raise(); } void Window::topWidget(QWidget *w) { _topWidget = w; } void Window::noTopWidget(QWidget *w) { if (_topWidget == w) { _topWidget = 0; } } void Window::showFromTray(QSystemTrayIcon::ActivationReason reason) { if (reason != QSystemTrayIcon::Context) { activate(); setWindowIcon(myIcon); psUpdateCounter(); if (App::main()) App::main()->setOnline(windowState()); } } void Window::toggleTray(QSystemTrayIcon::ActivationReason reason) { if (reason != QSystemTrayIcon::Context) { if (psIsActive()) { minimizeToTray(); } else { showFromTray(reason); } } } void Window::closeEvent(QCloseEvent *e) { if (MTP::authedId() && minimizeToTray()) { e->ignore(); } else { App::quit(); } } TitleWidget *Window::getTitle() { return title; } void Window::resizeEvent(QResizeEvent *e) { title->setGeometry(QRect(0, 0, width(), st::titleHeight + st::titleShadow)); if (layer) layer->resize(width(), height()); if (layerBG) layerBG->resize(width(), height()); if (_connecting) _connecting->setGeometry(0, height() - _connecting->height(), _connecting->width(), _connecting->height()); emit resized(QSize(width(), height() - st::titleHeight)); } Window::TempDirState Window::tempDirState() { if (_tempDeleter) { return TempDirRemoving; } return QDir(cTempDir()).exists() ? TempDirExists : TempDirEmpty; } void Window::tempDirDelete() { if (_tempDeleter) return; _tempDeleterThread = new QThread(); _tempDeleter = new TempDirDeleter(_tempDeleterThread); connect(_tempDeleter, SIGNAL(succeed()), this, SLOT(onTempDirCleared())); connect(_tempDeleter, SIGNAL(failed()), this, SLOT(onTempDirClearFailed())); _tempDeleterThread->start(); } void Window::onTempDirCleared() { _tempDeleter->deleteLater(); _tempDeleter = 0; _tempDeleterThread->deleteLater(); _tempDeleterThread = 0; emit tempDirCleared(); } void Window::onTempDirClearFailed() { _tempDeleter->deleteLater(); _tempDeleter = 0; _tempDeleterThread->deleteLater(); _tempDeleterThread = 0; emit tempDirClearFailed(); } void Window::notifySchedule(History *history, MsgId msgId) { if (App::quiting() || !history->currentNotification()) return; bool haveSetting = (history->peer->notify != UnknownNotifySettings); if (haveSetting) { if (history->peer->notify != EmptyNotifySettings && history->peer->notify->mute > unixtime()) { history->clearNotifications(); return; } } else { App::wnd()->getNotifySetting(MTP_inputNotifyPeer(history->peer->input)); } uint64 ms = getms() + NotifyWaitTimeout; notifyWhenAlerts[history].insert(ms); if (cDesktopNotify()) { NotifyWhenMaps::iterator i = notifyWhenMaps.find(history); if (i == notifyWhenMaps.end()) { i = notifyWhenMaps.insert(history, NotifyWhenMap()); } if (i.value().constFind(msgId) == i.value().cend()) { i.value().insert(msgId, ms); } NotifyWaiters *addTo = haveSetting ? ¬ifyWaiters : ¬ifySettingWaiters; if (addTo->constFind(history) == addTo->cend()) { addTo->insert(history, NotifyWaiter(msgId, ms)); } } if (haveSetting) { if (!notifyWaitTimer.isActive()) { notifyWaitTimer.start(NotifyWaitTimeout); } } } void Window::notifyFire() { notifyShowNext(); } void Window::notifyClear(History *history) { if (!history) { for (NotifyWindows::const_iterator i = notifyWindows.cbegin(), e = notifyWindows.cend(); i != e; ++i) { (*i)->unlinkHistory(); } psClearNotifies(); for (NotifyWhenMaps::const_iterator i = notifyWhenMaps.cbegin(), e = notifyWhenMaps.cend(); i != e; ++i) { i.key()->clearNotifications(); } notifyWaiters.clear(); notifySettingWaiters.clear(); notifyWhenMaps.clear(); return; } notifyWaiters.remove(history); notifySettingWaiters.remove(history); for (NotifyWindows::const_iterator i = notifyWindows.cbegin(), e = notifyWindows.cend(); i != e; ++i) { (*i)->unlinkHistory(history); } psClearNotifies(history->peer->id); notifyWhenMaps.remove(history); notifyWhenAlerts.remove(history); } void Window::notifyClearFast() { notifyWaiters.clear(); notifySettingWaiters.clear(); for (NotifyWindows::const_iterator i = notifyWindows.cbegin(), e = notifyWindows.cend(); i != e; ++i) { (*i)->deleteLater(); } psClearNotifies(); notifyWindows.clear(); notifyWhenMaps.clear(); notifyWhenAlerts.clear(); } void Window::notifySettingGot() { int32 t = unixtime(); for (NotifyWaiters::iterator i = notifySettingWaiters.begin(); i != notifySettingWaiters.end();) { History *history = i.key(); if (history->peer->notify == UnknownNotifySettings) { ++i; } else { if (history->peer->notify == EmptyNotifySettings || history->peer->notify->mute <= t) { notifyWaiters.insert(i.key(), i.value()); } i = notifySettingWaiters.erase(i); } } notifyWaitTimer.stop(); notifyShowNext(); } void Window::notifyShowNext(NotifyWindow *remove) { if (App::quiting()) return; int32 count = NotifyWindowsCount; if (remove) { for (NotifyWindows::iterator i = notifyWindows.begin(), e = notifyWindows.end(); i != e; ++i) { if ((*i) == remove) { notifyWindows.erase(i); break; } } } uint64 ms = getms(), nextAlert = 0; bool alert = false; for (NotifyWhenAlerts::iterator i = notifyWhenAlerts.begin(); i != notifyWhenAlerts.end();) { while (!i.value().isEmpty() && *i.value().begin() <= ms) { i.value().erase(i.value().begin()); NotifySettingsPtr n = i.key()->peer->notify; if (n == EmptyNotifySettings || (n != UnknownNotifySettings && n->mute <= unixtime())) { alert = true; } } if (i.value().isEmpty()) { i = notifyWhenAlerts.erase(i); } else { if (!nextAlert || nextAlert > *i.value().begin()) { nextAlert = *i.value().begin(); } ++i; } } if (alert) { psFlash(); App::playSound(); } if (cCustomNotifies()) { for (NotifyWindows::const_iterator i = notifyWindows.cbegin(), e = notifyWindows.cend(); i != e; ++i) { int32 ind = (*i)->index(); if (ind < 0) continue; --count; } } if (count <= 0 || !cDesktopNotify()) { if (nextAlert) { notifyWaitTimer.start(nextAlert - ms); } return; } QRect r = psDesktopRect(); int32 x = r.x() + r.width() - st::notifyWidth - st::notifyDeltaX, y = r.y() + r.height() - st::notifyHeight - st::notifyDeltaY; while (count > 0) { uint64 next = 0; HistoryItem *notifyItem = 0; NotifyWaiters::iterator notifyWaiter; for (NotifyWaiters::iterator i = notifyWaiters.begin(); i != notifyWaiters.end(); ++i) { History *history = i.key(); if (history->currentNotification() && history->currentNotification()->id != i.value().msg) { NotifyWhenMaps::iterator j = notifyWhenMaps.find(history); if (j == notifyWhenMaps.end()) { history->clearNotifications(); i = notifyWaiters.erase(i); continue; } do { NotifyWhenMap::const_iterator k = j.value().constFind(history->currentNotification()->id); if (k != j.value().cend()) { i.value().msg = k.key(); i.value().when = k.value(); break; } history->skipNotification(); } while (history->currentNotification()); } if (!history->currentNotification()) { notifyWhenMaps.remove(history); i = notifyWaiters.erase(i); continue; } uint64 when = i.value().when; if (!notifyItem || next > when) { next = when; notifyItem = history->currentNotification(); notifyWaiter = i; } } if (notifyItem) { if (next > ms) { if (nextAlert && nextAlert < next) { next = nextAlert; nextAlert = 0; } notifyWaitTimer.start(next - ms); break; } else { if (cCustomNotifies()) { NotifyWindow *notify = new NotifyWindow(notifyItem, x, y); notifyWindows.push_back(notify); psNotifyShown(notify); --count; } else { psPlatformNotify(notifyItem); } uint64 ms = getms(); History *history = notifyItem->history(); history->skipNotification(); NotifyWhenMaps::iterator j = notifyWhenMaps.find(history); if (j == notifyWhenMaps.end() || !history->currentNotification()) { history->clearNotifications(); notifyWaiters.erase(notifyWaiter); if (j != notifyWhenMaps.end()) notifyWhenMaps.erase(j); continue; } j.value().remove(notifyItem->id); do { NotifyWhenMap::const_iterator k = j.value().constFind(history->currentNotification()->id); if (k != j.value().cend()) { notifyWaiter.value().msg = k.key(); notifyWaiter.value().when = k.value(); break; } history->skipNotification(); } while (history->currentNotification()); if (!history->currentNotification()) { notifyWaiters.erase(notifyWaiter); notifyWhenMaps.erase(j); continue; } } } else { break; } } if (nextAlert) { notifyWaitTimer.start(nextAlert - ms); } count = NotifyWindowsCount - count; for (NotifyWindows::const_iterator i = notifyWindows.cbegin(), e = notifyWindows.cend(); i != e; ++i) { int32 ind = (*i)->index(); if (ind < 0) continue; --count; (*i)->moveTo(x, y - count * (st::notifyHeight + st::notifyDeltaY)); } } void Window::notifyItemRemoved(HistoryItem *item) { if (cCustomNotifies()) { for (NotifyWindows::const_iterator i = notifyWindows.cbegin(), e = notifyWindows.cend(); i != e; ++i) { (*i)->itemRemoved(item); } } } void Window::notifyStopHiding() { if (cCustomNotifies()) { for (NotifyWindows::const_iterator i = notifyWindows.cbegin(), e = notifyWindows.cend(); i != e; ++i) { (*i)->stopHiding(); } } } void Window::notifyStartHiding() { if (cCustomNotifies()) { for (NotifyWindows::const_iterator i = notifyWindows.cbegin(), e = notifyWindows.cend(); i != e; ++i) { (*i)->startHiding(); } } } void Window::notifyUpdateAllPhotos() { if (cCustomNotifies()) { for (NotifyWindows::const_iterator i = notifyWindows.cbegin(), e = notifyWindows.cend(); i != e; ++i) { (*i)->updatePeerPhoto(); } } } void Window::notifyUpdateAll() { if (cCustomNotifies()) { for (NotifyWindows::const_iterator i = notifyWindows.cbegin(), e = notifyWindows.cend(); i != e; ++i) { (*i)->updateNotifyDisplay(); } } psClearNotifies(); } void Window::notifyActivateAll() { if (cCustomNotifies()) { for (NotifyWindows::const_iterator i = notifyWindows.cbegin(), e = notifyWindows.cend(); i != e; ++i) { psActivateNotify(*i); } } } QImage Window::iconLarge() const { return icon256; } void Window::sendPaths() { if (settings) { hideSettings(); } else { if (layerShown()) { hideLayer(); } if (main && !main->animating()) { main->activate(); } } } Window::~Window() { notifyClearFast(); delete _tempDeleter; delete _tempDeleterThread; delete _connecting; delete trayIcon; delete trayIconMenu; delete intro; delete main; delete layer; delete settings; }