You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
qterminal-packaging/src/main.cpp

319 lines
9.5 KiB

/***************************************************************************
* Copyright (C) 2006 by Vladimir Kuznetsov *
* vovanec@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 2 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 <http://www.gnu.org/licenses/>. *
***************************************************************************/
#include <QApplication>
#include <QtGlobal>
#include <assert.h>
#include <stdio.h>
#include <getopt.h>
#include <stdlib.h>
#ifdef HAVE_QDBUS
#include <QtDBus/QtDBus>
#include <unistd.h>
#include "processadaptor.h"
#endif
#include "mainwindow.h"
#include "qterminalapp.h"
#include "terminalconfig.h"
#define out
const char* const short_options = "vhw:e:dp:";
const struct option long_options[] = {
{"version", 0, NULL, 'v'},
{"help", 0, NULL, 'h'},
{"workdir", 1, NULL, 'w'},
{"execute", 1, NULL, 'e'},
{"drop", 0, NULL, 'd'},
{"profile", 1, NULL, 'p'},
{NULL, 0, NULL, 0}
};
QTerminalApp * QTerminalApp::m_instance = NULL;
void print_usage_and_exit(int code)
{
printf("QTerminal %s\n", STR_VERSION);
puts("Usage: qterminal [OPTION]...\n");
puts(" -d, --drop Start in \"dropdown mode\" (like Yakuake or Tilda)");
puts(" -e, --execute <command> Execute command instead of shell");
puts(" -h, --help Print this help");
puts(" -p, --profile Load qterminal with specific options");
puts(" -v, --version Prints application version and exits");
puts(" -w, --workdir <dir> Start session with specified work directory");
puts("\nHomepage: <https://github.com/lxqt/qterminal>");
puts("Report bugs to <https://github.com/lxqt/qterminal/issues>");
exit(code);
}
void print_version_and_exit(int code=0)
{
printf("%s\n", STR_VERSION);
exit(code);
}
void parse_args(int argc, char* argv[], QString& workdir, QString & shell_command, out bool& dropMode)
{
int next_option;
dropMode = false;
do{
next_option = getopt_long(argc, argv, short_options, long_options, NULL);
switch(next_option)
{
case 'h':
print_usage_and_exit(0);
case 'w':
workdir = QString(optarg);
break;
case 'e':
shell_command = QString(optarg);
// #15 "Raw" -e params
// Passing "raw" params (like konsole -e mcedit /tmp/tmp.txt") is more preferable - then I can call QString("qterminal -e ") + cmd_line in other programs
while (optind < argc)
{
//printf("arg: %d - %s\n", optind, argv[optind]);
shell_command += ' ' + QString(argv[optind++]);
}
break;
case 'd':
dropMode = true;
break;
case 'p':
Properties::Instance(QString(optarg));
break;
case '?':
print_usage_and_exit(1);
case 'v':
print_version_and_exit();
}
}
while(next_option != -1);
}
int main(int argc, char *argv[])
{
QApplication::setApplicationName("qterminal");
QApplication::setApplicationVersion(STR_VERSION);
QApplication::setOrganizationDomain("qterminal.org");
#if QT_VERSION >= QT_VERSION_CHECK(5, 7, 0)
QApplication::setDesktopFileName(QLatin1String("qterminal.desktop"));
#endif
// Warning: do not change settings format. It can screw bookmarks later.
QSettings::setDefaultFormat(QSettings::IniFormat);
QTerminalApp *app = QTerminalApp::Instance(argc, argv);
#ifdef HAVE_QDBUS
app->registerOnDbus();
#endif
QString workdir, shell_command;
bool dropMode;
parse_args(argc, argv, workdir, shell_command, dropMode);
Properties::Instance()->migrate_settings();
Properties::Instance()->loadSettings();
qputenv("TERM", Properties::Instance()->term.toLatin1());
if (workdir.isEmpty())
workdir = QDir::currentPath();
app->setWorkingDirectory(workdir);
const QSettings settings;
const QFileInfo customStyle = QFileInfo(
QFileInfo(settings.fileName()).canonicalPath() +
"/style.qss"
);
if (customStyle.isFile() && customStyle.isReadable())
{
QFile style(customStyle.canonicalFilePath());
style.open(QFile::ReadOnly);
QString styleString = QLatin1String(style.readAll());
app->setStyleSheet(styleString);
}
// icons
/* setup our custom icon theme if there is no system theme (OS X, Windows) */
QCoreApplication::instance()->setAttribute(Qt::AA_UseHighDpiPixmaps); //Fix for High-DPI systems
if (QIcon::themeName().isEmpty())
QIcon::setThemeName("QTerminal");
// translations
QString fname = QString("qterminal_%1.qm").arg(QLocale::system().name().left(5));
QTranslator translator;
#ifdef TRANSLATIONS_DIR
qDebug() << "TRANSLATIONS_DIR: Loading translation file" << fname << "from dir" << TRANSLATIONS_DIR;
qDebug() << "load success:" << translator.load(fname, TRANSLATIONS_DIR, "_");
#endif
#ifdef APPLE_BUNDLE
qDebug() << "APPLE_BUNDLE: Loading translator file" << fname << "from dir" << QApplication::applicationDirPath()+"../translations";
qDebug() << "load success:" << translator.load(fname, QApplication::applicationDirPath()+"../translations", "_");
#endif
app->installTranslator(&translator);
TerminalConfig initConfig = TerminalConfig(workdir, shell_command);
app->newWindow(dropMode, initConfig);
int ret = app->exec();
delete Properties::Instance();
app->cleanup();
return ret;
}
MainWindow *QTerminalApp::newWindow(bool dropMode, TerminalConfig &cfg)
{
MainWindow *window = NULL;
if (dropMode)
{
QWidget *hiddenPreviewParent = new QWidget(0, Qt::Tool);
window = new MainWindow(cfg, dropMode, hiddenPreviewParent);
if (Properties::Instance()->dropShowOnStart)
window->show();
}
else
{
window = new MainWindow(cfg, dropMode);
window->show();
}
return window;
}
QTerminalApp *QTerminalApp::Instance()
{
assert(m_instance != NULL);
return m_instance;
}
QTerminalApp *QTerminalApp::Instance(int &argc, char **argv)
{
assert(m_instance == NULL);
m_instance = new QTerminalApp(argc, argv);
return m_instance;
}
QTerminalApp::QTerminalApp(int &argc, char **argv)
:QApplication(argc, argv)
{
}
QString &QTerminalApp::getWorkingDirectory()
{
return m_workDir;
}
void QTerminalApp::setWorkingDirectory(const QString &wd)
{
m_workDir = wd;
}
void QTerminalApp::cleanup() {
delete m_instance;
m_instance = NULL;
}
void QTerminalApp::addWindow(MainWindow *window)
{
m_windowList.append(window);
}
void QTerminalApp::removeWindow(MainWindow *window)
{
m_windowList.removeOne(window);
}
QList<MainWindow *> QTerminalApp::getWindowList()
{
return m_windowList;
}
#ifdef HAVE_QDBUS
void QTerminalApp::registerOnDbus()
{
if (!QDBusConnection::sessionBus().isConnected())
{
fprintf(stderr, "Cannot connect to the D-Bus session bus.\n"
"To start it, run:\n"
"\teval `dbus-launch --auto-syntax`\n");
return;
}
QString serviceName = QStringLiteral("org.lxqt.QTerminal-%1").arg(getpid());
if (!QDBusConnection::sessionBus().registerService(serviceName))
{
fprintf(stderr, "%s\n", qPrintable(QDBusConnection::sessionBus().lastError().message()));
return;
}
new ProcessAdaptor(this);
QDBusConnection::sessionBus().registerObject("/", this);
}
QList<QDBusObjectPath> QTerminalApp::getWindows()
{
QList<QDBusObjectPath> windows;
for (MainWindow *wnd : qAsConst(m_windowList))
{
windows.push_back(wnd->getDbusPath());
}
return windows;
}
QDBusObjectPath QTerminalApp::newWindow(const QHash<QString,QVariant> &termArgs)
{
TerminalConfig cfg = TerminalConfig::fromDbus(termArgs);
MainWindow *wnd = newWindow(false, cfg);
assert(wnd != NULL);
return wnd->getDbusPath();
}
QDBusObjectPath QTerminalApp::getActiveWindow()
{
QWidget *aw = activeWindow();
if (aw == NULL)
return QDBusObjectPath("/");
return qobject_cast<MainWindow*>(aw)->getDbusPath();
}
bool QTerminalApp::isDropMode() {
if (m_windowList.count() == 0) {
return false;
}
MainWindow *wnd = m_windowList.at(0);
return wnd->dropMode();
}
bool QTerminalApp::toggleDropdown() {
if (m_windowList.count() == 0) {
return false;
}
MainWindow *wnd = m_windowList.at(0);
if (!wnd->dropMode()) {
return false;
}
wnd->showHide();
return true;
}
#endif