mirror of
https://github.com/YACReader/yacreader
synced 2025-05-28 03:10:27 -04:00
Merge pull request #50 from selmf/mac_tray_icon
Minimize to tray, start to tray.
This commit is contained in:
commit
eab114aa66
@ -115,6 +115,7 @@ HEADERS += comic_flow.h \
|
||||
../common/pdf_comic.h \
|
||||
no_libraries_widget.h \
|
||||
import_widget.h \
|
||||
trayicon_controller.h \
|
||||
yacreader_local_server.h \
|
||||
yacreader_main_toolbar.h \
|
||||
comics_remover.h \
|
||||
@ -184,6 +185,7 @@ SOURCES += comic_flow.cpp \
|
||||
../common/onstart_flow_selection_dialog.cpp \
|
||||
no_libraries_widget.cpp \
|
||||
import_widget.cpp \
|
||||
trayicon_controller.cpp \
|
||||
yacreader_local_server.cpp \
|
||||
yacreader_main_toolbar.cpp \
|
||||
comics_remover.cpp \
|
||||
@ -221,6 +223,11 @@ SOURCES += comic_flow.cpp \
|
||||
SOURCES += ../common/gl/yacreader_flow_gl.cpp
|
||||
}
|
||||
|
||||
macx {
|
||||
HEADERS += trayhandler.h
|
||||
OBJECTIVE_SOURCES += trayhandler.mm
|
||||
}
|
||||
|
||||
include(./server/server.pri)
|
||||
include(../custom_widgets/custom_widgets_yacreaderlibrary.pri)
|
||||
|
||||
|
@ -71,5 +71,6 @@
|
||||
<file alias="images/sidebar/delete_sidebar@2x.png">../images/sidebar/delete_sidebar_osx@2x.png</file>
|
||||
<file alias="images/sidebar/addLabelIcon@2x.png">../images/sidebar/addLabelIcon_osx@2x.png</file>
|
||||
<file alias="images/sidebar/renameListIcon@2x.png">../images/sidebar/renameListIcon_osx@2x.png</file>
|
||||
<file>macostrayicon.svg</file>
|
||||
</qresource>
|
||||
</RCC>
|
||||
|
@ -41,5 +41,6 @@
|
||||
<file>../images/lists/label_yellow.png</file>
|
||||
<file>../images/lists/list.png</file>
|
||||
<file>../images/empty_reading_list.png</file>
|
||||
<file>icon.ico</file>
|
||||
</qresource>
|
||||
</RCC>
|
||||
|
@ -81,12 +81,16 @@
|
||||
|
||||
#include "yacreader_comics_views_manager.h"
|
||||
|
||||
#include "trayicon_controller.h"
|
||||
|
||||
#include "QsLog.h"
|
||||
|
||||
#ifdef Q_OS_WIN
|
||||
#include <shellapi.h>
|
||||
#endif
|
||||
|
||||
using namespace YACReader;
|
||||
|
||||
LibraryWindow::LibraryWindow()
|
||||
: QMainWindow(), fullscreen(false), previousFilter(""), fetching(false), status(LibraryWindow::Normal), removeError(false)
|
||||
{
|
||||
@ -136,37 +140,26 @@ void LibraryWindow::setupUI()
|
||||
else
|
||||
//if(settings->value(USE_OPEN_GL).toBool() == false)
|
||||
showMaximized();
|
||||
}
|
||||
/* //disabled until icons are ready and macos native code is done
|
||||
trayIcon.setIcon(QApplication::windowIcon());
|
||||
trayIcon.show();
|
||||
connect(&trayIcon, SIGNAL(activated(QSystemTrayIcon::ActivationReason)),
|
||||
this, SLOT(trayActivation(QSystemTrayIcon::ActivationReason)));
|
||||
}
|
||||
|
||||
trayIconController = new TrayIconController(settings, this);
|
||||
}
|
||||
|
||||
void LibraryWindow::trayActivation(QSystemTrayIcon::ActivationReason reason)
|
||||
/*void LibraryWindow::changeEvent(QEvent *event)
|
||||
{
|
||||
if (reason == QSystemTrayIcon::Trigger)
|
||||
{
|
||||
setWindowState((windowState() & ~Qt::WindowMinimized));
|
||||
show();
|
||||
activateWindow();
|
||||
raise();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void LibraryWindow::changeEvent(QEvent *event)
|
||||
{
|
||||
if (event->type() == QEvent::WindowStateChange && isMinimized())
|
||||
{
|
||||
hide();
|
||||
}
|
||||
else
|
||||
{
|
||||
QMainWindow::changeEvent(event);
|
||||
}
|
||||
|
||||
if (event->type() == QEvent::WindowStateChange && isMinimized() &&
|
||||
trayIcon.isVisible()) {
|
||||
#ifdef Q_OS_MACOS
|
||||
OSXHideDockIcon();
|
||||
#endif
|
||||
hide();
|
||||
} else if (event->type() == QEvent::WindowStateChange) {
|
||||
#ifdef Q_OS_MACOS
|
||||
OSXShowDockIcon();
|
||||
#endif
|
||||
show();
|
||||
}
|
||||
}*/
|
||||
|
||||
void LibraryWindow::doLayout()
|
||||
@ -2281,8 +2274,9 @@ void LibraryWindow::importLibrary(QString clc, QString destPath, QString name)
|
||||
|
||||
void LibraryWindow::reloadOptions()
|
||||
{
|
||||
//comicFlow->setFlowType(flowType);
|
||||
comicsViewsManager->comicsView->updateConfig(settings);
|
||||
|
||||
trayIconController->updateIconVisibility();
|
||||
}
|
||||
|
||||
QString LibraryWindow::currentPath()
|
||||
@ -2318,6 +2312,14 @@ void LibraryWindow::showImportComicsInfo()
|
||||
#include "startup.h"
|
||||
extern Startup *s;
|
||||
void LibraryWindow::closeEvent(QCloseEvent *event)
|
||||
{
|
||||
if (!trayIconController->handleCloseToTrayIcon(event)) {
|
||||
event->accept();
|
||||
closeApp();
|
||||
}
|
||||
}
|
||||
|
||||
void LibraryWindow::prepareToCloseApp()
|
||||
{
|
||||
s->stop();
|
||||
settings->setValue(MAIN_WINDOW_GEOMETRY, saveGeometry());
|
||||
@ -2326,8 +2328,13 @@ void LibraryWindow::closeEvent(QCloseEvent *event)
|
||||
sideBar->close();
|
||||
|
||||
QApplication::instance()->processEvents();
|
||||
event->accept();
|
||||
QMainWindow::closeEvent(event);
|
||||
}
|
||||
|
||||
void LibraryWindow::closeApp()
|
||||
{
|
||||
prepareToCloseApp();
|
||||
|
||||
qApp->exit(0);
|
||||
}
|
||||
|
||||
void LibraryWindow::showNoLibrariesWidget()
|
||||
@ -2506,14 +2513,14 @@ void LibraryWindow::showFoldersContextMenu(const QPoint &point)
|
||||
/*
|
||||
void LibraryWindow::showSocial()
|
||||
{
|
||||
socialDialog->move(this->mapToGlobal(QPoint(width()-socialDialog->width()-10, centralWidget()->pos().y()+10)));
|
||||
socialDialog->move(this->mapToGlobal(QPoint(width()-socialDialog->width()-10, centralWidget()->pos().y()+10)));
|
||||
|
||||
QModelIndexList indexList = getSelectedComics();
|
||||
QModelIndexList indexList = getSelectedComics();
|
||||
|
||||
ComicDB comic = dmCV->getComic(indexList.at(0));
|
||||
ComicDB comic = dmCV->getComic(indexList.at(0));
|
||||
|
||||
socialDialog->setComic(comic,currentPath());
|
||||
socialDialog->setHidden(false);
|
||||
socialDialog->setComic(comic,currentPath());
|
||||
socialDialog->setHidden(false);
|
||||
}*/
|
||||
|
||||
void LibraryWindow::libraryAlreadyExists(const QString &name)
|
||||
|
@ -5,7 +5,6 @@
|
||||
#include <QMap>
|
||||
#include <QModelIndex>
|
||||
#include <QFileInfo>
|
||||
#include <QSystemTrayIcon>
|
||||
|
||||
#include "yacreader_global_gui.h"
|
||||
#include "yacreader_libraries.h"
|
||||
@ -77,6 +76,10 @@ class EmptySpecialListWidget;
|
||||
class EmptyReadingListWidget;
|
||||
class YACReaderComicsViewsManager;
|
||||
|
||||
namespace YACReader {
|
||||
class TrayIconController;
|
||||
}
|
||||
|
||||
#include "comic_db.h"
|
||||
|
||||
using namespace YACReader;
|
||||
@ -385,7 +388,8 @@ public slots:
|
||||
void checkMaxNumLibraries();
|
||||
void showErrorUpgradingLibrary(const QString &path);
|
||||
|
||||
//void changeEvent(QEvent *event);
|
||||
void prepareToCloseApp();
|
||||
void closeApp();
|
||||
|
||||
private:
|
||||
//fullscreen mode in Windows for preventing this bug: QTBUG-41309 https://bugreports.qt.io/browse/QTBUG-41309
|
||||
@ -393,10 +397,8 @@ private:
|
||||
QPoint previousPos;
|
||||
QSize previousSize;
|
||||
std::future<void> upgradeLibraryFuture;
|
||||
QSystemTrayIcon trayIcon;
|
||||
|
||||
private slots:
|
||||
//void trayActivation(QSystemTrayIcon::ActivationReason reason);
|
||||
TrayIconController *trayIconController;
|
||||
};
|
||||
|
||||
#endif
|
||||
|
BIN
YACReaderLibrary/macostrayicon.svg
Normal file
BIN
YACReaderLibrary/macostrayicon.svg
Normal file
Binary file not shown.
After Width: | Height: | Size: 2.3 KiB |
@ -20,6 +20,9 @@
|
||||
#include "yacreader_libraries.h"
|
||||
#include "exit_check.h"
|
||||
#include "opengl_checker.h"
|
||||
#ifdef Q_OS_MACOS
|
||||
#include "trayhandler.h"
|
||||
#endif
|
||||
|
||||
#include "QsLog.h"
|
||||
#include "QsLogDest.h"
|
||||
@ -127,9 +130,12 @@ int main(int argc, char **argv)
|
||||
app.setApplicationVersion(VERSION);
|
||||
app.setAttribute(Qt::AA_UseHighDpiPixmaps);
|
||||
|
||||
// Set window icon according to Freedesktop icon specification
|
||||
// This is mostly relevant for Linux and other Unix systems
|
||||
if (QIcon::hasThemeIcon("YACReaderLibrary")) {
|
||||
app.setWindowIcon(QIcon::fromTheme("YACReaderLibrary"));
|
||||
}
|
||||
// TODO: We might want to add a fallback icon here.
|
||||
|
||||
QString destLog = YACReader::getSettingsPath() + "/yacreaderlibrary.log";
|
||||
QDir().mkpath(YACReader::getSettingsPath());
|
||||
@ -239,7 +245,15 @@ int main(int argc, char **argv)
|
||||
|
||||
//connections to localServer
|
||||
|
||||
mw->show();
|
||||
// start as tray
|
||||
if (!settings->value(START_TO_TRAY, false).toBool() || !settings->value(CLOSE_TO_TRAY, false).toBool()) {
|
||||
mw->show();
|
||||
}
|
||||
#ifdef Q_OS_MACOS
|
||||
else {
|
||||
OSXHideDockIcon();
|
||||
}
|
||||
#endif
|
||||
|
||||
int ret = app.exec();
|
||||
|
||||
|
@ -43,6 +43,27 @@ OptionsDialog::OptionsDialog(QWidget *parent)
|
||||
#ifndef NO_OPENGL
|
||||
sw->hide();
|
||||
#endif
|
||||
// Tray icon settings
|
||||
QGroupBox *trayIconBox = new QGroupBox(tr("Tray icon settings (experimental)"));
|
||||
QVBoxLayout *trayLayout = new QVBoxLayout();
|
||||
|
||||
trayIconCheckbox = new QCheckBox(tr("Close to tray"));
|
||||
startToTrayCheckbox = new QCheckBox(tr("Start into the system tray"));
|
||||
|
||||
connect(trayIconCheckbox, &QCheckBox::clicked,
|
||||
[=](bool checked) {
|
||||
settings->setValue(CLOSE_TO_TRAY, checked);
|
||||
startToTrayCheckbox->setEnabled(checked);
|
||||
emit optionsChanged();
|
||||
});
|
||||
connect(startToTrayCheckbox, &QCheckBox::clicked,
|
||||
[=](bool checked) {
|
||||
settings->setValue(START_TO_TRAY, checked);
|
||||
});
|
||||
|
||||
trayLayout->addWidget(trayIconCheckbox);
|
||||
trayLayout->addWidget(startToTrayCheckbox);
|
||||
trayIconBox->setLayout(trayLayout);
|
||||
|
||||
auto apiKeyLayout = new QVBoxLayout();
|
||||
auto apiKeyButton = new QPushButton(tr("Edit Comic Vine API key"));
|
||||
@ -115,6 +136,7 @@ OptionsDialog::OptionsDialog(QWidget *parent)
|
||||
|
||||
auto generalW = new QWidget;
|
||||
generalW->setLayout(generalLayout);
|
||||
generalLayout->addWidget(trayIconBox);
|
||||
generalLayout->addWidget(shortcutsBox);
|
||||
generalLayout->addWidget(apiKeyBox);
|
||||
generalLayout->addStretch();
|
||||
@ -146,6 +168,10 @@ void OptionsDialog::restoreOptions(QSettings *settings)
|
||||
{
|
||||
YACReaderOptionsDialog::restoreOptions(settings);
|
||||
|
||||
trayIconCheckbox->setChecked(settings->value(CLOSE_TO_TRAY, false).toBool());
|
||||
startToTrayCheckbox->setChecked(settings->value(START_TO_TRAY, false).toBool());
|
||||
startToTrayCheckbox->setEnabled(trayIconCheckbox->isChecked());
|
||||
|
||||
bool useBackgroundImage = settings->value(USE_BACKGROUND_IMAGE_IN_GRID_VIEW, true).toBool();
|
||||
|
||||
useBackgroundImageCheck->setChecked(useBackgroundImage);
|
||||
|
@ -34,8 +34,9 @@ private:
|
||||
QLabel *opacityLabel;
|
||||
QLabel *blurLabel;
|
||||
QPushButton *resetButton;
|
||||
|
||||
QCheckBox *displayContinueReadingBannerCheck;
|
||||
QCheckBox *trayIconCheckbox;
|
||||
QCheckBox *startToTrayCheckbox;
|
||||
};
|
||||
|
||||
#endif
|
||||
|
7
YACReaderLibrary/trayhandler.h
Normal file
7
YACReaderLibrary/trayhandler.h
Normal file
@ -0,0 +1,7 @@
|
||||
#ifndef TRAY_HANDLER
|
||||
#define TRAY_HANDLER
|
||||
|
||||
void OSXShowDockIcon();
|
||||
void OSXHideDockIcon();
|
||||
|
||||
#endif
|
11
YACReaderLibrary/trayhandler.mm
Normal file
11
YACReaderLibrary/trayhandler.mm
Normal file
@ -0,0 +1,11 @@
|
||||
#import <AppKit/AppKit.h>
|
||||
#include "trayhandler.h"
|
||||
|
||||
void OSXShowDockIcon()
|
||||
{
|
||||
[NSApp setActivationPolicy:NSApplicationActivationPolicyRegular];
|
||||
}
|
||||
void OSXHideDockIcon()
|
||||
{
|
||||
[NSApp setActivationPolicy:NSApplicationActivationPolicyAccessory];
|
||||
}
|
102
YACReaderLibrary/trayicon_controller.cpp
Normal file
102
YACReaderLibrary/trayicon_controller.cpp
Normal file
@ -0,0 +1,102 @@
|
||||
#include "trayicon_controller.h"
|
||||
|
||||
#include "yacreader_global_gui.h"
|
||||
|
||||
#include "library_window.h"
|
||||
|
||||
#include <QtWidgets>
|
||||
#include <QMessageBox>
|
||||
|
||||
#ifdef Q_OS_MACOS
|
||||
#include "trayhandler.h"
|
||||
#endif
|
||||
|
||||
using namespace YACReader;
|
||||
|
||||
TrayIconController::TrayIconController(QSettings *settings, LibraryWindow *window)
|
||||
: QObject(nullptr), settings(settings), window(window)
|
||||
{
|
||||
|
||||
// If a window icon was set in main() we reuse it for the tray too.
|
||||
// This allows support for third party icon themes on Freedesktop(Linux/Unix)
|
||||
// systems.
|
||||
if (!QApplication::windowIcon().isNull()) {
|
||||
trayIcon.setIcon(QApplication::windowIcon());
|
||||
} else {
|
||||
#ifdef Q_OS_WIN
|
||||
trayIcon.setIcon(QIcon(":/icon.ico"));
|
||||
#else
|
||||
#ifdef Q_OS_MACOS
|
||||
auto icon = QIcon(":/macostrayicon.svg");
|
||||
icon.setIsMask(true);
|
||||
trayIcon.setIcon(icon);
|
||||
#else
|
||||
trayIcon.setIcon(QIcon(":/images/iconLibrary.png"));
|
||||
#endif
|
||||
#endif
|
||||
}
|
||||
|
||||
connect(&trayIcon, &QSystemTrayIcon::activated,
|
||||
[=](QSystemTrayIcon::ActivationReason reason) {
|
||||
if (reason == QSystemTrayIcon::DoubleClick) {
|
||||
showWindow();
|
||||
}
|
||||
});
|
||||
|
||||
auto restoreAction = new QAction(tr("&Restore"), this);
|
||||
connect(restoreAction, &QAction::triggered, this, &TrayIconController::showWindow);
|
||||
|
||||
auto quitAction = new QAction(tr("&Quit"), this);
|
||||
connect(quitAction, &QAction::triggered, window, &LibraryWindow::closeApp);
|
||||
|
||||
trayIconMenu = new QMenu(this->window);
|
||||
trayIconMenu->addAction(restoreAction);
|
||||
trayIconMenu->addSeparator();
|
||||
trayIconMenu->addAction(quitAction);
|
||||
|
||||
trayIcon.setContextMenu(trayIconMenu);
|
||||
|
||||
updateIconVisibility();
|
||||
}
|
||||
|
||||
void TrayIconController::updateIconVisibility()
|
||||
{
|
||||
trayIcon.setVisible(settings->value(CLOSE_TO_TRAY, false).toBool());
|
||||
}
|
||||
|
||||
bool TrayIconController::handleCloseToTrayIcon(QCloseEvent *event)
|
||||
{
|
||||
#ifdef Q_OS_OSX
|
||||
if (!event->spontaneous() || !window->isVisible()) {
|
||||
return false;
|
||||
}
|
||||
#endif
|
||||
if (trayIcon.isVisible()) {
|
||||
if (!settings->value(CLOSE_TO_TRAY_NOTIFIED, false).toBool()) {
|
||||
QMessageBox::information(window, tr("Systray"),
|
||||
tr("YACReaderLibrary will keep running in the "
|
||||
"system tray. To terminate the program, "
|
||||
"choose <b>Quit</b> in the context menu "
|
||||
"of the system tray icon."));
|
||||
settings->setValue(CLOSE_TO_TRAY_NOTIFIED, true);
|
||||
}
|
||||
#ifdef Q_OS_OSX
|
||||
OSXHideDockIcon();
|
||||
#endif
|
||||
window->hide();
|
||||
event->ignore();
|
||||
return true;
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
void TrayIconController::showWindow()
|
||||
{
|
||||
#ifdef Q_OS_MACOS
|
||||
OSXShowDockIcon();
|
||||
#endif
|
||||
window->showNormal();
|
||||
window->raise(); // for MacOS
|
||||
window->activateWindow(); // for Windows
|
||||
}
|
36
YACReaderLibrary/trayicon_controller.h
Normal file
36
YACReaderLibrary/trayicon_controller.h
Normal file
@ -0,0 +1,36 @@
|
||||
#ifndef TRAYICON_CONTROLLER_H
|
||||
#define TRAYICON_CONTROLLER_H
|
||||
|
||||
#include <QtWidgets>
|
||||
#include <QSystemTrayIcon>
|
||||
|
||||
class LibraryWindow;
|
||||
|
||||
namespace YACReader {
|
||||
|
||||
class TrayIconController : public QObject
|
||||
{
|
||||
Q_OBJECT
|
||||
public:
|
||||
TrayIconController(QSettings *settings,
|
||||
LibraryWindow *window);
|
||||
|
||||
void updateIconVisibility();
|
||||
|
||||
bool handleCloseToTrayIcon(QCloseEvent *event);
|
||||
|
||||
QSystemTrayIcon trayIcon;
|
||||
|
||||
public slots:
|
||||
void showWindow();
|
||||
|
||||
private:
|
||||
QSettings *settings;
|
||||
LibraryWindow *window;
|
||||
|
||||
QMenu *trayIconMenu;
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
#endif // TRAYICON_CONTROLLER_H
|
@ -17,6 +17,9 @@
|
||||
#define FULLSCREEN "FULLSCREEN"
|
||||
#define Y_WINDOW_GEOMETRY "GEOMETRY"
|
||||
#define MAXIMIZED "MAXIMIZED"
|
||||
#define CLOSE_TO_TRAY "CLOSE_TO_TRAY"
|
||||
#define CLOSE_TO_TRAY_NOTIFIED "CLOSE_TO_TRAY_NOTIFIED"
|
||||
#define START_TO_TRAY "START_TO_TRAY"
|
||||
#define DOUBLE_PAGE "DOUBLE_PAGE"
|
||||
#define DOUBLE_MANGA_PAGE "DOUBLE_MANGA_PAGE"
|
||||
#define BACKGROUND_COLOR "BACKGROUND_COLOR"
|
||||
|
Loading…
Reference in New Issue
Block a user