mirror of
https://github.com/fergalmoran/flameshot.git
synced 2025-12-22 09:51:06 +00:00
509 lines
16 KiB
C++
509 lines
16 KiB
C++
// Copyright(c) 2017-2019 Alejandro Sirgo Rica & Contributors
|
|
//
|
|
// This file is part of Flameshot.
|
|
//
|
|
// Flameshot 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.
|
|
//
|
|
// Flameshot 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.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with Flameshot. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
#include "controller.h"
|
|
#include "src/config/configwindow.h"
|
|
#include "src/utils/confighandler.h"
|
|
#include "src/utils/history.h"
|
|
#include "src/utils/screengrabber.h"
|
|
#include "src/utils/systemnotification.h"
|
|
#include "src/widgets/capture/capturetoolbutton.h"
|
|
#include "src/widgets/capture/capturewidget.h"
|
|
#include "src/widgets/capturelauncher.h"
|
|
#include "src/widgets/historywidget.h"
|
|
#include "src/widgets/infowindow.h"
|
|
#include "src/widgets/notificationwidget.h"
|
|
#include <QAction>
|
|
#include <QApplication>
|
|
#include <QClipboard>
|
|
#include <QDesktopServices>
|
|
#include <QDesktopWidget>
|
|
#include <QJsonDocument>
|
|
#include <QJsonObject>
|
|
#include <QMenu>
|
|
#include <QNetworkAccessManager>
|
|
#include <QNetworkReply>
|
|
#include <QNetworkRequest>
|
|
#include <QSystemTrayIcon>
|
|
|
|
#ifdef Q_OS_WIN
|
|
#include "src/core/globalshortcutfilter.h"
|
|
#endif
|
|
|
|
#if (defined(Q_OS_MAC) || defined(Q_OS_MAC64) || defined(Q_OS_MACOS) || \
|
|
defined(Q_OS_MACX))
|
|
#include <QOperatingSystemVersion>
|
|
#include <QScreen>
|
|
#endif
|
|
|
|
// Controller is the core component of Flameshot, creates the trayIcon and
|
|
// launches the capture widget
|
|
|
|
Controller::Controller()
|
|
: m_captureWindow(nullptr)
|
|
, m_history(nullptr)
|
|
, m_trayIconMenu(nullptr)
|
|
, m_networkCheckUpdates(nullptr)
|
|
, m_showCheckAppUpdateStatus(false)
|
|
{
|
|
m_appLatestVersion = QStringLiteral(APP_VERSION).replace("v", "");
|
|
qApp->setQuitOnLastWindowClosed(false);
|
|
|
|
// set default shortcusts if not set yet
|
|
ConfigHandler().setShortcutsDefault();
|
|
|
|
// init tray icon
|
|
#if defined(Q_OS_LINUX) || defined(Q_OS_UNIX)
|
|
if (!ConfigHandler().disabledTrayIconValue()) {
|
|
enableTrayIcon();
|
|
}
|
|
#elif defined(Q_OS_WIN)
|
|
enableTrayIcon();
|
|
|
|
GlobalShortcutFilter* nativeFilter = new GlobalShortcutFilter(this);
|
|
qApp->installNativeEventFilter(nativeFilter);
|
|
connect(nativeFilter, &GlobalShortcutFilter::printPressed, this, [this]() {
|
|
this->requestCapture(CaptureRequest(CaptureRequest::GRAPHICAL_MODE));
|
|
});
|
|
#endif
|
|
|
|
QString StyleSheet = CaptureButton::globalStyleSheet();
|
|
qApp->setStyleSheet(StyleSheet);
|
|
|
|
#if (defined(Q_OS_MAC) || defined(Q_OS_MAC64) || defined(Q_OS_MACOS) || \
|
|
defined(Q_OS_MACX))
|
|
// Try to take a test screenshot, MacOS will request a "Screen Recording"
|
|
// permissions on the first run. Otherwise it will be hidden under the
|
|
// CaptureWidget
|
|
QScreen* currentScreen = QGuiApplication::screenAt(QCursor::pos());
|
|
currentScreen->grabWindow(QApplication::desktop()->winId(), 0, 0, 1, 1);
|
|
#endif
|
|
getLatestAvailableVersion();
|
|
}
|
|
|
|
Controller::~Controller()
|
|
{
|
|
delete m_history;
|
|
delete m_trayIconMenu;
|
|
}
|
|
|
|
Controller* Controller::getInstance()
|
|
{
|
|
static Controller c;
|
|
return &c;
|
|
}
|
|
|
|
void Controller::enableExports()
|
|
{
|
|
connect(
|
|
this, &Controller::captureTaken, this, &Controller::handleCaptureTaken);
|
|
connect(
|
|
this, &Controller::captureFailed, this, &Controller::handleCaptureFailed);
|
|
}
|
|
|
|
void Controller::getLatestAvailableVersion()
|
|
{
|
|
// This features is required for MacOS and Windows user and for Linux users
|
|
// who installed Flameshot not from the repository.
|
|
m_networkCheckUpdates = new QNetworkAccessManager();
|
|
m_networkCheckUpdates = new QNetworkAccessManager(this);
|
|
QNetworkRequest requestCheckUpdates(QUrl(FLAMESHOT_APP_VERSION_URL));
|
|
connect(m_networkCheckUpdates,
|
|
&QNetworkAccessManager::finished,
|
|
this,
|
|
&Controller::handleReplyCheckUpdates);
|
|
m_networkCheckUpdates->get(requestCheckUpdates);
|
|
}
|
|
|
|
void Controller::handleReplyCheckUpdates(QNetworkReply* reply)
|
|
{
|
|
if (reply->error() == QNetworkReply::NoError) {
|
|
QJsonDocument response = QJsonDocument::fromJson(reply->readAll());
|
|
QJsonObject json = response.object();
|
|
m_appLatestVersion = json["tag_name"].toString().replace("v", "");
|
|
if (m_appLatestVersion.compare(
|
|
QStringLiteral(APP_VERSION).replace("v", "")) < 0) {
|
|
// Next commented lines are for debugging
|
|
// if (m_appLatestVersion.compare(
|
|
// QStringLiteral("v0.8.5.4").replace("v", "")) > 0) {
|
|
m_appLatestUrl = json["html_url"].toString();
|
|
QString newVersion =
|
|
tr("New version %1 is available").arg(m_appLatestVersion);
|
|
m_appUpdates->setText(newVersion);
|
|
if (m_showCheckAppUpdateStatus) {
|
|
sendTrayNotification(newVersion, "Flameshot", 5);
|
|
QDesktopServices::openUrl(QUrl(m_appLatestUrl));
|
|
}
|
|
} else if (m_showCheckAppUpdateStatus) {
|
|
sendTrayNotification(
|
|
tr("You have the latest version"), "Flameshot", 5);
|
|
}
|
|
}
|
|
// nothing to do on fails, is not critical for checking for updates
|
|
m_showCheckAppUpdateStatus = false;
|
|
}
|
|
|
|
void Controller::appUpdates()
|
|
{
|
|
if (m_appLatestUrl.isEmpty()) {
|
|
m_showCheckAppUpdateStatus = true;
|
|
getLatestAvailableVersion();
|
|
} else {
|
|
QDesktopServices::openUrl(QUrl(m_appLatestUrl));
|
|
}
|
|
}
|
|
|
|
void Controller::requestCapture(const CaptureRequest& request)
|
|
{
|
|
uint id = request.id();
|
|
m_requestMap.insert(id, request);
|
|
|
|
switch (request.captureMode()) {
|
|
case CaptureRequest::FULLSCREEN_MODE:
|
|
doLater(request.delay(), this, [this, id]() {
|
|
this->startFullscreenCapture(id);
|
|
});
|
|
break;
|
|
// TODO: Figure out the code path that gets here so the deprated warning
|
|
// can be fixed
|
|
case CaptureRequest::SCREEN_MODE: {
|
|
int&& number = request.data().toInt();
|
|
doLater(request.delay(), this, [this, id, number]() {
|
|
this->startScreenGrab(id, number);
|
|
});
|
|
break;
|
|
}
|
|
case CaptureRequest::GRAPHICAL_MODE: {
|
|
QString&& path = request.path();
|
|
doLater(request.delay(), this, [this, id, path]() {
|
|
this->startVisualCapture(id, path);
|
|
});
|
|
break;
|
|
}
|
|
default:
|
|
emit captureFailed(id);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// creation of a new capture in GUI mode
|
|
void Controller::startVisualCapture(const uint id,
|
|
const QString& forcedSavePath)
|
|
{
|
|
if (!m_captureWindow) {
|
|
QWidget* modalWidget = nullptr;
|
|
do {
|
|
modalWidget = qApp->activeModalWidget();
|
|
if (modalWidget) {
|
|
modalWidget->close();
|
|
modalWidget->deleteLater();
|
|
}
|
|
} while (modalWidget);
|
|
|
|
m_captureWindow = new CaptureWidget(id, forcedSavePath);
|
|
// m_captureWindow = new CaptureWidget(id, forcedSavePath, false); //
|
|
// debug
|
|
connect(m_captureWindow,
|
|
&CaptureWidget::captureFailed,
|
|
this,
|
|
&Controller::captureFailed);
|
|
connect(m_captureWindow,
|
|
&CaptureWidget::captureTaken,
|
|
this,
|
|
&Controller::captureTaken);
|
|
|
|
#ifdef Q_OS_WIN
|
|
m_captureWindow->show();
|
|
#elif (defined(Q_OS_MAC) || defined(Q_OS_MAC64) || defined(Q_OS_MACOS) || \
|
|
defined(Q_OS_MACX))
|
|
// In "Emulate fullscreen mode"
|
|
m_captureWindow->showFullScreen();
|
|
m_captureWindow->activateWindow();
|
|
m_captureWindow->raise();
|
|
#else
|
|
m_captureWindow->showFullScreen();
|
|
#endif
|
|
if (!m_appLatestUrl.isEmpty() &&
|
|
0 != m_appLatestVersion.compare(
|
|
ConfigHandler().ignoreUpdateToVersion())) {
|
|
m_captureWindow->showAppUpdateNotification(m_appLatestVersion,
|
|
m_appLatestUrl);
|
|
}
|
|
} else {
|
|
emit captureFailed(id);
|
|
}
|
|
}
|
|
|
|
void Controller::startScreenGrab(const uint id, const int screenNumber)
|
|
{
|
|
bool ok = true;
|
|
int n = screenNumber;
|
|
|
|
if (n < 0) {
|
|
QPoint globalCursorPos = QCursor::pos();
|
|
n = qApp->desktop()->screenNumber(globalCursorPos);
|
|
}
|
|
QPixmap p(ScreenGrabber().grabScreen(n, ok));
|
|
if (ok) {
|
|
emit captureTaken(id, p);
|
|
} else {
|
|
emit captureFailed(id);
|
|
}
|
|
}
|
|
|
|
// creation of the configuration window
|
|
void Controller::openConfigWindow()
|
|
{
|
|
if (!m_configWindow) {
|
|
m_configWindow = new ConfigWindow();
|
|
m_configWindow->show();
|
|
#if (defined(Q_OS_MAC) || defined(Q_OS_MAC64) || defined(Q_OS_MACOS) || \
|
|
defined(Q_OS_MACX))
|
|
m_configWindow->activateWindow();
|
|
m_configWindow->raise();
|
|
#endif
|
|
}
|
|
}
|
|
|
|
// creation of the window of information
|
|
void Controller::openInfoWindow()
|
|
{
|
|
if (!m_infoWindow) {
|
|
m_infoWindow = new InfoWindow();
|
|
#if (defined(Q_OS_MAC) || defined(Q_OS_MAC64) || defined(Q_OS_MACOS) || \
|
|
defined(Q_OS_MACX))
|
|
m_infoWindow->activateWindow();
|
|
m_infoWindow->raise();
|
|
#endif
|
|
}
|
|
}
|
|
|
|
void Controller::openLauncherWindow()
|
|
{
|
|
if (!m_launcherWindow) {
|
|
m_launcherWindow = new CaptureLauncher();
|
|
}
|
|
m_launcherWindow->show();
|
|
#if (defined(Q_OS_MAC) || defined(Q_OS_MAC64) || defined(Q_OS_MACOS) || \
|
|
defined(Q_OS_MACX))
|
|
m_launcherWindow->activateWindow();
|
|
m_launcherWindow->raise();
|
|
#endif
|
|
}
|
|
|
|
void Controller::enableTrayIcon()
|
|
{
|
|
if (m_trayIcon) {
|
|
return;
|
|
}
|
|
m_trayIconMenu = new QMenu();
|
|
|
|
ConfigHandler().setDisabledTrayIcon(false);
|
|
QAction* captureAction = new QAction(tr("&Take Screenshot"), this);
|
|
connect(captureAction, &QAction::triggered, this, [this]() {
|
|
// Wait 400 ms to hide the QMenu
|
|
doLater(400, this, [this]() { this->startVisualCapture(); });
|
|
});
|
|
QAction* launcherAction = new QAction(tr("&Open Launcher"), this);
|
|
connect(launcherAction,
|
|
&QAction::triggered,
|
|
this,
|
|
&Controller::openLauncherWindow);
|
|
QAction* configAction = new QAction(tr("&Configuration"), this);
|
|
connect(
|
|
configAction, &QAction::triggered, this, &Controller::openConfigWindow);
|
|
QAction* infoAction = new QAction(tr("&About"), this);
|
|
connect(infoAction, &QAction::triggered, this, &Controller::openInfoWindow);
|
|
|
|
m_appUpdates = new QAction(tr("Check for updates"), this);
|
|
connect(m_appUpdates, &QAction::triggered, this, &Controller::appUpdates);
|
|
|
|
QAction* quitAction = new QAction(tr("&Quit"), this);
|
|
connect(quitAction, &QAction::triggered, qApp, &QCoreApplication::quit);
|
|
|
|
// recent screenshots
|
|
QAction* recentAction = new QAction(tr("&Latest Uploads"), this);
|
|
connect(
|
|
recentAction, SIGNAL(triggered()), this, SLOT(showRecentScreenshots()));
|
|
|
|
// generate menu
|
|
m_trayIconMenu->addAction(captureAction);
|
|
m_trayIconMenu->addAction(launcherAction);
|
|
m_trayIconMenu->addSeparator();
|
|
m_trayIconMenu->addAction(recentAction);
|
|
m_trayIconMenu->addSeparator();
|
|
m_trayIconMenu->addAction(configAction);
|
|
m_trayIconMenu->addSeparator();
|
|
m_trayIconMenu->addAction(m_appUpdates);
|
|
m_trayIconMenu->addAction(infoAction);
|
|
m_trayIconMenu->addSeparator();
|
|
m_trayIconMenu->addAction(quitAction);
|
|
|
|
m_trayIcon = new QSystemTrayIcon();
|
|
m_trayIcon->setToolTip(QStringLiteral("Flameshot"));
|
|
#if defined(Q_OS_MAC) || defined(Q_OS_MAC64) || defined(Q_OS_MACOS) || \
|
|
defined(Q_OS_MACX)
|
|
// Because of the following issues on MacOS "Catalina":
|
|
// https://bugreports.qt.io/browse/QTBUG-86393
|
|
// https://developer.apple.com/forums/thread/126072
|
|
auto currentMacOsVersion = QOperatingSystemVersion::current();
|
|
if (currentMacOsVersion >= currentMacOsVersion.MacOSBigSur) {
|
|
m_trayIcon->setContextMenu(m_trayIconMenu);
|
|
}
|
|
#else
|
|
m_trayIcon->setContextMenu(m_trayIconMenu);
|
|
#endif
|
|
QIcon trayicon =
|
|
QIcon::fromTheme("flameshot-tray", QIcon(":img/app/flameshot.png"));
|
|
m_trayIcon->setIcon(trayicon);
|
|
|
|
#if defined(Q_OS_MAC) || defined(Q_OS_MAC64) || defined(Q_OS_MACOS) || \
|
|
defined(Q_OS_MACX)
|
|
if (currentMacOsVersion < currentMacOsVersion.MacOSBigSur) {
|
|
// Because of the following issues on MacOS "Catalina":
|
|
// https://bugreports.qt.io/browse/QTBUG-86393
|
|
// https://developer.apple.com/forums/thread/126072
|
|
auto trayIconActivated = [this](QSystemTrayIcon::ActivationReason r) {
|
|
if (m_trayIconMenu->isVisible()) {
|
|
m_trayIconMenu->hide();
|
|
} else {
|
|
m_trayIconMenu->popup(QCursor::pos());
|
|
}
|
|
};
|
|
connect(
|
|
m_trayIcon, &QSystemTrayIcon::activated, this, trayIconActivated);
|
|
}
|
|
#else
|
|
auto trayIconActivated = [this](QSystemTrayIcon::ActivationReason r) {
|
|
if (r == QSystemTrayIcon::Trigger) {
|
|
startVisualCapture();
|
|
}
|
|
};
|
|
connect(m_trayIcon, &QSystemTrayIcon::activated, this, trayIconActivated);
|
|
#endif
|
|
|
|
#ifdef Q_OS_WIN
|
|
// Ensure proper removal of tray icon when program quits on Windows.
|
|
connect(
|
|
qApp, &QCoreApplication::aboutToQuit, m_trayIcon, &QSystemTrayIcon::hide);
|
|
#endif
|
|
|
|
m_trayIcon->show();
|
|
if (ConfigHandler().showStartupLaunchMessage()) {
|
|
m_trayIcon->showMessage(
|
|
"Flameshot",
|
|
QObject::tr(
|
|
"Hello, I'm here! Click icon in the tray to take a screenshot or "
|
|
"click with a right button to see more options."),
|
|
QSystemTrayIcon::Information,
|
|
3000);
|
|
}
|
|
}
|
|
|
|
void Controller::disableTrayIcon()
|
|
{
|
|
#if defined(Q_OS_LINUX) || defined(Q_OS_UNIX)
|
|
if (m_trayIcon) {
|
|
m_trayIcon->deleteLater();
|
|
}
|
|
ConfigHandler().setDisabledTrayIcon(true);
|
|
#endif
|
|
}
|
|
|
|
void Controller::sendTrayNotification(const QString& text,
|
|
const QString& title,
|
|
const int timeout)
|
|
{
|
|
if (m_trayIcon) {
|
|
m_trayIcon->showMessage(
|
|
title, text, QIcon(":img/app/flameshot.svg"), timeout);
|
|
}
|
|
}
|
|
|
|
void Controller::updateConfigComponents()
|
|
{
|
|
if (m_configWindow) {
|
|
m_configWindow->updateChildren();
|
|
}
|
|
}
|
|
|
|
void Controller::updateRecentScreenshots()
|
|
{
|
|
if (nullptr != m_history) {
|
|
if (m_history->isVisible()) {
|
|
m_history->loadHistory();
|
|
}
|
|
}
|
|
}
|
|
|
|
void Controller::showRecentScreenshots()
|
|
{
|
|
if (nullptr == m_history) {
|
|
m_history = new HistoryWidget();
|
|
}
|
|
m_history->loadHistory();
|
|
m_history->show();
|
|
#if (defined(Q_OS_MAC) || defined(Q_OS_MAC64) || defined(Q_OS_MACOS) || \
|
|
defined(Q_OS_MACX))
|
|
m_history->activateWindow();
|
|
m_history->raise();
|
|
#endif
|
|
}
|
|
|
|
void Controller::startFullscreenCapture(const uint id)
|
|
{
|
|
bool ok = true;
|
|
QPixmap p(ScreenGrabber().grabEntireDesktop(ok));
|
|
if (ok) {
|
|
emit captureTaken(id, p);
|
|
} else {
|
|
emit captureFailed(id);
|
|
}
|
|
}
|
|
|
|
void Controller::handleCaptureTaken(uint id, QPixmap p)
|
|
{
|
|
auto it = m_requestMap.find(id);
|
|
if (it != m_requestMap.end()) {
|
|
it.value().exportCapture(p);
|
|
m_requestMap.erase(it);
|
|
}
|
|
if (ConfigHandler().closeAfterScreenshotValue()) {
|
|
QApplication::quit();
|
|
}
|
|
}
|
|
|
|
void Controller::handleCaptureFailed(uint id)
|
|
{
|
|
m_requestMap.remove(id);
|
|
|
|
if (ConfigHandler().closeAfterScreenshotValue()) {
|
|
QApplication::quit();
|
|
}
|
|
}
|
|
|
|
void Controller::doLater(int msec, QObject* receiver, lambda func)
|
|
{
|
|
QTimer* timer = new QTimer(receiver);
|
|
QObject::connect(timer, &QTimer::timeout, receiver, [timer, func]() {
|
|
func();
|
|
timer->deleteLater();
|
|
});
|
|
timer->setInterval(msec);
|
|
timer->start();
|
|
}
|