462 lines
15 KiB
C++
462 lines
15 KiB
C++
/*
|
|
* Copyright (C) 2020-2022 Roy Qu (royqh1979@gmail.com)
|
|
*
|
|
* This program 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.
|
|
*
|
|
* This program 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 this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
#include "mainwindow.h"
|
|
#include "settings.h"
|
|
#include "systemconsts.h"
|
|
#include "utils.h"
|
|
#include <QApplication>
|
|
#include <QDir>
|
|
#include <QTranslator>
|
|
#include <QStandardPaths>
|
|
#include <QMessageBox>
|
|
#include <QStringList>
|
|
#include <QAbstractNativeEventFilter>
|
|
#include <QDesktopWidget>
|
|
#include <QDir>
|
|
#include <QScreen>
|
|
#include <QLockFile>
|
|
#include "common.h"
|
|
#include "colorscheme.h"
|
|
#include "iconsmanager.h"
|
|
#include "autolinkmanager.h"
|
|
#include <qt_utils/charsetinfo.h>
|
|
#include "parser/parserutils.h"
|
|
#include "editorlist.h"
|
|
#include "widgets/choosethemedialog.h"
|
|
#include "thememanager.h"
|
|
|
|
#ifdef Q_OS_WIN
|
|
#include <QTemporaryFile>
|
|
#include <windows.h>
|
|
#include <psapi.h>
|
|
#include <QSharedMemory>
|
|
#include <QBuffer>
|
|
#include <winuser.h>
|
|
|
|
#include "widgets/cpudialog.h"
|
|
#endif
|
|
|
|
QString getSettingFilename(const QString& filepath, bool& firstRun);
|
|
#ifdef Q_OS_WIN
|
|
class WindowLogoutEventFilter : public QAbstractNativeEventFilter {
|
|
|
|
// QAbstractNativeEventFilter interface
|
|
public:
|
|
bool nativeEventFilter(const QByteArray &eventType, void *message, long *result) override;
|
|
};
|
|
|
|
#ifndef WM_DPICHANGED
|
|
# define WM_DPICHANGED 0x02e0
|
|
#endif
|
|
|
|
#define WM_APP_OPEN_FILE (WM_APP + 6736 /* “OPEN” on dial pad */)
|
|
static_assert(WM_APP_OPEN_FILE < 0xc000);
|
|
|
|
HWND prevAppInstance = NULL;
|
|
BOOL CALLBACK GetPreviousInstanceCallback(HWND hwnd, LPARAM param){
|
|
BOOL result = TRUE;
|
|
WCHAR buffer[4098];
|
|
HINSTANCE hWindowModule = (HINSTANCE)GetWindowLongPtr(hwnd,GWLP_HINSTANCE);
|
|
|
|
if (hWindowModule==0)
|
|
return result;
|
|
|
|
DWORD processID;
|
|
|
|
// Get the ID of the process that created this window
|
|
GetWindowThreadProcessId(hwnd,&processID);
|
|
if (processID==0)
|
|
return result;
|
|
|
|
// Get the process associated with the ID
|
|
HANDLE hWindowProcess = OpenProcess(PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, FALSE, processID);
|
|
if (hWindowProcess == 0)
|
|
return result;
|
|
|
|
// Get its module filename
|
|
if (GetModuleFileNameExW(hWindowProcess, hWindowModule, buffer, sizeof(buffer)) == 0)
|
|
return TRUE;
|
|
|
|
CloseHandle(hWindowProcess); // not needed anymore
|
|
WCHAR * compareFilename=(WCHAR*)param;
|
|
QString s1=QString::fromWCharArray(compareFilename);
|
|
QString s2=QString::fromWCharArray(buffer);
|
|
|
|
//Is from the "same" application?
|
|
if (QString::compare(s1,s2,PATH_SENSITIVITY)==0) {
|
|
//found, stop EnumWindows loop
|
|
prevAppInstance = hwnd;
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
HWND getPreviousInstance() {
|
|
WCHAR buffer[4098];
|
|
//ShowMessage('ERROR_ALREADY_EXISTS');
|
|
// Store our own module filename
|
|
if (GetModuleFileNameW(GetModuleHandle(NULL), buffer, sizeof(buffer)) == 0)
|
|
return NULL;
|
|
|
|
// If that's the case, walk all top level windows and find the previous instance
|
|
// At this point, the program that created the mutex might not have created its MainForm yet
|
|
if (EnumWindows(GetPreviousInstanceCallback, LPARAM(buffer))==0) {
|
|
return prevAppInstance;
|
|
} else
|
|
return NULL;
|
|
}
|
|
|
|
bool WindowLogoutEventFilter::nativeEventFilter(const QByteArray & /*eventType*/, void *message, long *result){
|
|
MSG * pMsg = static_cast<MSG *>(message);
|
|
switch(pMsg->message) {
|
|
case WM_QUERYENDSESSION:
|
|
if (pMsg->lParam == 0
|
|
|| (pMsg->lParam & ENDSESSION_LOGOFF)) {
|
|
if (!pMainWindow->close()) {
|
|
*result = 0;
|
|
} else {
|
|
*result = 1;
|
|
}
|
|
return true;
|
|
}
|
|
break;
|
|
case WM_DPICHANGED:{
|
|
if (pMsg->hwnd == (HWND)pMainWindow->winId()) {
|
|
int oldDPI = screenDPI();
|
|
//postEvent takes the owner ship
|
|
QEvent * dpiEvent = new QEvent(DPI_CHANGED_EVENT);
|
|
qApp->postEvent(pMainWindow,dpiEvent);
|
|
setScreenDPI(HIWORD(pMsg->wParam));
|
|
int newDPI = screenDPI();
|
|
pMainWindow->updateDPI(oldDPI,newDPI);
|
|
} else if (pMainWindow->cpuDialog() &&
|
|
(HWND)pMainWindow->cpuDialog()->winId() == pMsg->hwnd) {
|
|
int newDPI = HIWORD(pMsg->wParam);
|
|
pMainWindow->cpuDialog()->updateDPI(newDPI);
|
|
}
|
|
break;
|
|
}
|
|
case WM_APP_OPEN_FILE: {
|
|
QSharedMemory sharedMemory("RedPandaCpp/openfiles");
|
|
if (sharedMemory.attach()) {
|
|
QBuffer buffer;
|
|
QDataStream in(&buffer);
|
|
QStringList files;
|
|
sharedMemory.lock();
|
|
buffer.setData((char*)sharedMemory.constData(), sharedMemory.size());
|
|
buffer.open(QBuffer::ReadOnly);
|
|
in >> files;
|
|
sharedMemory.unlock();
|
|
if (pMainWindow->isMinimized()) {
|
|
pMainWindow->showNormal();
|
|
}
|
|
pMainWindow->openFiles(files);
|
|
sharedMemory.detach();
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool sendFilesToInstance() {
|
|
HWND prevInstance = getPreviousInstance();
|
|
if (prevInstance != NULL) {
|
|
QSharedMemory sharedMemory("RedPandaCpp/openfiles");
|
|
QBuffer buffer;
|
|
buffer.open(QBuffer::ReadWrite);
|
|
QDataStream out(&buffer);
|
|
QStringList filesToOpen = qApp->arguments();
|
|
filesToOpen.pop_front();
|
|
out<<filesToOpen;
|
|
int size = buffer.size();
|
|
if (sharedMemory.create(size)) {
|
|
sharedMemory.lock();
|
|
char *to = (char*)sharedMemory.data();
|
|
const char *from = buffer.data().data();
|
|
memcpy(to, from, qMin(sharedMemory.size(), size));
|
|
sharedMemory.unlock();
|
|
SendMessage(prevInstance,WM_APP_OPEN_FILE,0,0);
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
QString getSettingFilename(const QString& filepath, bool& firstRun) {
|
|
QString filename;
|
|
if (filepath.isEmpty()) {
|
|
if (isGreenEdition()) {
|
|
filename = includeTrailingPathDelimiter(QApplication::applicationDirPath()) +
|
|
"config/" + APP_SETTSINGS_FILENAME;
|
|
} else {
|
|
filename =includeTrailingPathDelimiter(QStandardPaths::standardLocations(QStandardPaths::AppDataLocation)[0])
|
|
+ APP_SETTSINGS_FILENAME;
|
|
}
|
|
} else {
|
|
filename = filepath;
|
|
}
|
|
|
|
QFileInfo fileInfo(filename);
|
|
firstRun = !fileInfo.exists();
|
|
QDir dir(fileInfo.absoluteDir());
|
|
if (!dir.exists()) {
|
|
if (!dir.mkpath(dir.absolutePath())) {
|
|
QMessageBox::critical(nullptr, QObject::tr("Error"),
|
|
QString(QObject::tr("Can't create configuration folder %1")).arg(dir.absolutePath()));
|
|
return "";
|
|
}
|
|
}
|
|
|
|
if (fileInfo.exists() && !fileInfo.isWritable()) {
|
|
QMessageBox::critical(nullptr, QObject::tr("Error"),
|
|
QString(QObject::tr("Can't write to configuration file %1")).arg(filename));
|
|
|
|
return "";
|
|
}
|
|
return filename;
|
|
}
|
|
|
|
void setTheme(const QString& theme) {
|
|
pSettings->environment().setTheme(theme);
|
|
ThemeManager themeManager;
|
|
PAppTheme appTheme = themeManager.theme(theme);
|
|
if (appTheme && !appTheme->defaultColorScheme().isEmpty()) {
|
|
pSettings->editor().setColorScheme(appTheme->defaultColorScheme());
|
|
pSettings->editor().save();
|
|
}
|
|
if (appTheme && !appTheme->defaultIconSet().isEmpty()) {
|
|
pSettings->environment().setIconSet(appTheme->defaultIconSet());
|
|
}
|
|
pSettings->environment().save();
|
|
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
#ifdef Q_OS_WINDOWS
|
|
// Make title bar and palette follow system-wide dark mode setting on recent Windows releases.
|
|
// Use freetype as the fontengine
|
|
qputenv("QT_QPA_PLATFORM", "windows:darkmode=2:fontengine=freetype");
|
|
#endif
|
|
|
|
#ifdef Q_OS_MACOS
|
|
// in macOS GUI apps, `/usr/local/bin` is not in PATH by default
|
|
// follow the Unix way by prepending it to `/usr/bin`
|
|
{
|
|
QStringList pathList = getExecutableSearchPaths();
|
|
if (!pathList.contains("/usr/local/bin")) {
|
|
auto idxUsrBin = pathList.indexOf("/usr/bin");
|
|
if (idxUsrBin >= 0)
|
|
pathList.insert(idxUsrBin, "/usr/local/bin");
|
|
else
|
|
pathList.append("/usr/local/bin");
|
|
}
|
|
QString newPath = pathList.join(PATH_SEPARATOR);
|
|
qputenv("PATH", newPath.toUtf8());
|
|
}
|
|
#endif
|
|
|
|
QApplication app(argc, argv);
|
|
|
|
app.setAttribute(Qt::AA_UseHighDpiPixmaps);
|
|
|
|
QLockFile lockFile(QDir::tempPath()+QDir::separator()+"RedPandaDevCppStartUp.lock");
|
|
{
|
|
bool firstRun;
|
|
QString settingFilename = getSettingFilename(QString(), firstRun);
|
|
bool openInSingleInstance = false;
|
|
if (!settingFilename.isEmpty() && !firstRun) {
|
|
QSettings envSetting(settingFilename,QSettings::IniFormat);
|
|
envSetting.beginGroup(SETTING_ENVIRONMENT);
|
|
openInSingleInstance = envSetting.value("open_files_in_single_instance",false).toBool();
|
|
} else if (!settingFilename.isEmpty() && firstRun)
|
|
openInSingleInstance = false;
|
|
if (app.arguments().contains("-ns")) {
|
|
openInSingleInstance = false;
|
|
} else if (app.arguments().contains("-s"))
|
|
openInSingleInstance = true;
|
|
if (openInSingleInstance) {
|
|
int openCount = 0;
|
|
while (true) {
|
|
if (lockFile.tryLock())
|
|
break;
|
|
QThread::msleep(100);
|
|
openCount++;
|
|
if (openCount>100)
|
|
break;
|
|
}
|
|
|
|
if (app.arguments().length()>=2 && openCount<100) {
|
|
#ifdef Q_OS_WIN
|
|
if (sendFilesToInstance()) {
|
|
lockFile.unlock();
|
|
return 0;
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
//Translation must be loaded first
|
|
QTranslator trans,transQt,transUtils;
|
|
bool firstRun;
|
|
QString settingFilename = getSettingFilename(QString(), firstRun);
|
|
if (!isGreenEdition()) {
|
|
QDir::setCurrent(QStandardPaths::standardLocations(QStandardPaths::DocumentsLocation)[0]);
|
|
}
|
|
if (settingFilename.isEmpty()) {
|
|
lockFile.unlock();
|
|
return -1;
|
|
}
|
|
QString language;
|
|
{
|
|
QSettings languageSetting(settingFilename,QSettings::IniFormat);
|
|
languageSetting.beginGroup(SETTING_ENVIRONMENT);
|
|
language = languageSetting.value("language",QLocale::system().name()).toString();
|
|
|
|
if (trans.load("RedPandaIDE_"+language,":/i18n/")) {
|
|
app.installTranslator(&trans);
|
|
}
|
|
if (transUtils.load("qt_utils_"+language,":/i18n/")) {
|
|
app.installTranslator(&transUtils);
|
|
}
|
|
if (transQt.load("qt_"+language,":/translations")) {
|
|
app.installTranslator(&transQt);
|
|
}
|
|
}
|
|
qRegisterMetaType<PCompileIssue>("PCompileIssue");
|
|
qRegisterMetaType<PCompileIssue>("PCompileIssue&");
|
|
qRegisterMetaType<QVector<int>>("QVector<int>");
|
|
qRegisterMetaType<QHash<int,QString>>("QHash<int,QString>");
|
|
|
|
initParser();
|
|
|
|
try {
|
|
|
|
SystemConsts systemConsts;
|
|
pSystemConsts = &systemConsts;
|
|
CharsetInfoManager charsetInfoManager(language);
|
|
pCharsetInfoManager=&charsetInfoManager;
|
|
|
|
//We must use smarter point here, to manually control it's lifetime:
|
|
// when restore default settings, it must be destoyed before we remove all setting files.
|
|
auto settings = std::make_unique<Settings>(settingFilename);
|
|
//load settings
|
|
pSettings = settings.get();
|
|
if (firstRun) {
|
|
pSettings->compilerSets().findSets();
|
|
pSettings->compilerSets().saveSets();
|
|
}
|
|
pSettings->load();
|
|
if (firstRun) {
|
|
//set theme
|
|
ChooseThemeDialog themeDialog;
|
|
#ifdef Q_OS_WINDOWS
|
|
// Qt's default style on Windows is not good.
|
|
themeDialog.hideAutoFollowSystemTheme();
|
|
#endif
|
|
themeDialog.exec();
|
|
switch (themeDialog.theme()) {
|
|
case ChooseThemeDialog::Theme::AutoFollowSystem:
|
|
setTheme("system");
|
|
break;
|
|
case ChooseThemeDialog::Theme::Dark:
|
|
setTheme("dark");
|
|
break;
|
|
case ChooseThemeDialog::Theme::Light:
|
|
setTheme("default");
|
|
break;
|
|
default:
|
|
setTheme("default");
|
|
}
|
|
|
|
pSettings->editor().setDefaultFileCpp(themeDialog.language()==ChooseThemeDialog::Language::CPlusPlus);
|
|
pSettings->editor().save();
|
|
|
|
//auto detect git in path
|
|
#ifdef ENABLE_VCS
|
|
pSettings->vcs().detectGitInPath();
|
|
#endif
|
|
}
|
|
//Color scheme settings must be loaded after translation
|
|
ColorManager colorManager;
|
|
pColorManager = &colorManager;
|
|
IconsManager iconsManager;
|
|
pIconsManager = &iconsManager;
|
|
AutolinkManager autolinkManager;
|
|
pAutolinkManager = &autolinkManager;
|
|
try {
|
|
pAutolinkManager->load();
|
|
} catch (FileError e) {
|
|
QMessageBox::critical(nullptr,
|
|
QObject::tr("Can't load autolink settings"),
|
|
e.reason(),
|
|
QMessageBox::Ok);
|
|
}
|
|
|
|
MainWindow mainWindow;
|
|
pMainWindow = &mainWindow;
|
|
#if QT_VERSION_MAJOR==5 && QT_VERSION_MINOR < 15
|
|
setScreenDPI(qApp->primaryScreen()->logicalDotsPerInch());
|
|
#else
|
|
if (mainWindow.screen())
|
|
setScreenDPI(mainWindow.screen()->logicalDotsPerInch());
|
|
#endif
|
|
mainWindow.show();
|
|
if (app.arguments().count()>1) {
|
|
QStringList filesToOpen = app.arguments();
|
|
filesToOpen.pop_front();
|
|
pMainWindow->openFiles(filesToOpen);
|
|
} else {
|
|
if (pSettings->editor().autoLoadLastFiles())
|
|
pMainWindow->loadLastOpens();
|
|
if (pMainWindow->editorList()->pageCount()==0 && !pMainWindow->project()) {
|
|
pMainWindow->newEditor();
|
|
}
|
|
}
|
|
|
|
//reset default open folder
|
|
QDir::setCurrent(pSettings->environment().defaultOpenFolder());
|
|
|
|
pMainWindow->setFilesViewRoot(pSettings->environment().currentFolder());
|
|
|
|
#ifdef Q_OS_WIN
|
|
WindowLogoutEventFilter filter;
|
|
app.installNativeEventFilter(&filter);
|
|
#endif
|
|
if (lockFile.isLocked()) {
|
|
lockFile.unlock();
|
|
}
|
|
|
|
int retCode = app.exec();
|
|
if (mainWindow.shouldRemoveAllSettings()) {
|
|
QString configDir = pSettings->dirs().config();
|
|
settings.release();
|
|
delete pSettings;
|
|
QDir dir(configDir);
|
|
dir.removeRecursively();
|
|
}
|
|
return retCode;
|
|
} catch (BaseError e) {
|
|
lockFile.unlock();
|
|
QMessageBox::critical(nullptr,QApplication::tr("Error"),e.reason());
|
|
return -1;
|
|
}
|
|
}
|