2022-12-03 22:27:49 +01:00
|
|
|
#include <QApplication>
|
2023-02-03 19:41:28 +01:00
|
|
|
#include <QEvent>
|
|
|
|
#include <QObject>
|
|
|
|
#include <QPushButton>
|
2022-12-03 22:27:49 +01:00
|
|
|
#include <QScreen>
|
|
|
|
#include <QTabWidget>
|
2023-02-03 19:41:28 +01:00
|
|
|
#include <QWidget>
|
2022-12-03 22:27:49 +01:00
|
|
|
#include <QtDebug>
|
2023-02-03 19:41:28 +01:00
|
|
|
#include <QtWidgets>
|
|
|
|
#include <array>
|
2022-12-03 22:27:49 +01:00
|
|
|
#include <cstdio>
|
2023-02-03 19:41:28 +01:00
|
|
|
#include <iostream>
|
2022-12-03 22:27:49 +01:00
|
|
|
#include <memory>
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <string>
|
2022-12-12 15:51:37 +01:00
|
|
|
#include <vector>
|
2022-12-03 22:27:49 +01:00
|
|
|
|
2023-02-03 19:41:28 +01:00
|
|
|
#include "mainpicker.h"
|
|
|
|
|
2022-12-03 22:27:49 +01:00
|
|
|
std::string execAndGet(const char* cmd) {
|
2023-09-06 20:36:48 +02:00
|
|
|
std::array<char, 128> buffer;
|
|
|
|
std::string result;
|
2022-12-03 22:27:49 +01:00
|
|
|
std::unique_ptr<FILE, decltype(&pclose)> pipe(popen(cmd, "r"), pclose);
|
|
|
|
if (!pipe) {
|
|
|
|
throw std::runtime_error("popen() failed!");
|
|
|
|
}
|
|
|
|
while (fgets(buffer.data(), buffer.size(), pipe.get()) != nullptr) {
|
|
|
|
result += buffer.data();
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-09-06 20:36:48 +02:00
|
|
|
QApplication* pickerPtr = nullptr;
|
|
|
|
MainPicker* mainPickerPtr = nullptr;
|
2022-12-12 15:51:37 +01:00
|
|
|
|
|
|
|
struct SWindowEntry {
|
2023-09-06 20:36:48 +02:00
|
|
|
std::string name;
|
|
|
|
std::string clazz;
|
2023-02-04 22:10:46 +01:00
|
|
|
unsigned long long id = 0;
|
2022-12-12 15:51:37 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
std::vector<SWindowEntry> getWindows(const char* env) {
|
|
|
|
std::vector<SWindowEntry> result;
|
|
|
|
|
|
|
|
if (!env)
|
|
|
|
return result;
|
|
|
|
|
|
|
|
std::string rolling = env;
|
|
|
|
|
|
|
|
while (!rolling.empty()) {
|
|
|
|
// ID
|
2023-02-03 19:41:28 +01:00
|
|
|
const auto IDSEPPOS = rolling.find("[HC>]");
|
2023-09-06 20:36:48 +02:00
|
|
|
const auto IDSTR = rolling.substr(0, IDSEPPOS);
|
2022-12-12 15:51:37 +01:00
|
|
|
|
|
|
|
// class
|
2023-02-03 19:41:28 +01:00
|
|
|
const auto CLASSSEPPOS = rolling.find("[HT>]");
|
2023-09-06 20:36:48 +02:00
|
|
|
const auto CLASSSTR = rolling.substr(IDSEPPOS + 5, CLASSSEPPOS - IDSEPPOS - 5);
|
2022-12-12 15:51:37 +01:00
|
|
|
|
|
|
|
// title
|
2023-02-03 19:41:28 +01:00
|
|
|
const auto TITLESEPPOS = rolling.find("[HE>]");
|
2023-09-06 20:36:48 +02:00
|
|
|
const auto TITLESTR = rolling.substr(CLASSSEPPOS + 5, TITLESEPPOS - 5 - CLASSSEPPOS);
|
2022-12-12 15:51:37 +01:00
|
|
|
|
|
|
|
try {
|
2023-02-04 22:10:46 +01:00
|
|
|
result.push_back({TITLESTR, CLASSSTR, std::stoull(IDSTR)});
|
2023-02-03 20:05:59 +01:00
|
|
|
} catch (std::exception& e) {
|
2022-12-17 19:00:42 +01:00
|
|
|
// silent err
|
2022-12-12 15:51:37 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
rolling = rolling.substr(TITLESEPPOS + 5);
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2022-12-03 22:27:49 +01:00
|
|
|
|
2023-02-03 19:41:28 +01:00
|
|
|
int main(int argc, char* argv[]) {
|
2022-12-03 22:27:49 +01:00
|
|
|
qputenv("QT_LOGGING_RULES", "qml=false");
|
2022-12-12 15:51:37 +01:00
|
|
|
|
2023-09-06 20:36:48 +02:00
|
|
|
const char* WINDOWLISTSTR = getenv("XDPH_WINDOW_SHARING_LIST");
|
2022-12-12 15:51:37 +01:00
|
|
|
|
2023-09-06 20:36:48 +02:00
|
|
|
const auto WINDOWLIST = getWindows(WINDOWLISTSTR);
|
2022-12-12 15:51:37 +01:00
|
|
|
|
2022-12-03 22:27:49 +01:00
|
|
|
QApplication picker(argc, argv);
|
|
|
|
pickerPtr = &picker;
|
|
|
|
MainPicker w;
|
2022-12-12 15:51:37 +01:00
|
|
|
mainPickerPtr = &w;
|
2022-12-03 22:27:49 +01:00
|
|
|
|
|
|
|
// get the tabwidget
|
2023-09-06 20:36:48 +02:00
|
|
|
const auto TABWIDGET = (QTabWidget*)w.children()[1]->children()[0];
|
|
|
|
const auto ALLOWTOKENBUTTON = (QCheckBox*)w.children()[1]->children()[1];
|
2022-12-03 22:27:49 +01:00
|
|
|
|
|
|
|
const auto TAB1 = (QWidget*)TABWIDGET->children()[0];
|
|
|
|
|
2023-09-06 20:36:48 +02:00
|
|
|
const auto SCREENS_SCROLL_AREA_CONTENTS =
|
|
|
|
(QWidget*)TAB1->findChild<QWidget*>("screens")->findChild<QScrollArea*>("scrollArea")->findChild<QWidget*>("scrollAreaWidgetContents");
|
2022-12-03 22:27:49 +01:00
|
|
|
|
|
|
|
// add all screens
|
2023-09-06 20:36:48 +02:00
|
|
|
const auto SCREENS = picker.screens();
|
2022-12-03 22:27:49 +01:00
|
|
|
|
2023-09-06 20:36:48 +02:00
|
|
|
constexpr int BUTTON_WIDTH = 441;
|
2022-12-03 22:27:49 +01:00
|
|
|
constexpr int BUTTON_HEIGHT = 41;
|
2023-09-06 20:36:48 +02:00
|
|
|
constexpr int BUTTON_PAD = 4;
|
2022-12-03 22:27:49 +01:00
|
|
|
|
|
|
|
for (int i = 0; i < SCREENS.size(); ++i) {
|
2023-09-06 20:36:48 +02:00
|
|
|
const auto GEOMETRY = SCREENS[i]->geometry();
|
2022-12-03 22:27:49 +01:00
|
|
|
|
2023-09-06 20:36:48 +02:00
|
|
|
QString text = QString::fromStdString(std::string("Screen " + std::to_string(i) + " at " + std::to_string(GEOMETRY.x()) + ", " + std::to_string(GEOMETRY.y()) + " (" +
|
|
|
|
std::to_string(GEOMETRY.width()) + "x" + std::to_string(GEOMETRY.height()) + ") (") +
|
|
|
|
SCREENS[i]->name().toStdString() + ")");
|
2022-12-03 22:27:49 +01:00
|
|
|
QPushButton* button = new QPushButton(text, (QWidget*)SCREENS_SCROLL_AREA_CONTENTS);
|
|
|
|
button->move(9, 5 + (BUTTON_HEIGHT + BUTTON_PAD) * i);
|
|
|
|
button->resize(BUTTON_WIDTH, BUTTON_HEIGHT);
|
2023-02-03 19:41:28 +01:00
|
|
|
QObject::connect(button, &QPushButton::clicked, [=]() {
|
2022-12-03 22:27:49 +01:00
|
|
|
std::string ID = button->text().toStdString();
|
2023-09-06 20:36:48 +02:00
|
|
|
ID = ID.substr(ID.find_last_of('(') + 1);
|
|
|
|
ID = ID.substr(0, ID.find_last_of(')'));
|
|
|
|
|
|
|
|
std::cout << (ALLOWTOKENBUTTON->isChecked() ? "r" : "");
|
|
|
|
std::cout << "/";
|
2022-12-03 22:27:49 +01:00
|
|
|
|
|
|
|
std::cout << "screen:" << ID << "\n";
|
|
|
|
pickerPtr->quit();
|
2022-12-04 22:12:57 +01:00
|
|
|
return 0;
|
2022-12-03 22:27:49 +01:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
SCREENS_SCROLL_AREA_CONTENTS->resize(SCREENS_SCROLL_AREA_CONTENTS->size().width(), 5 + (BUTTON_HEIGHT + BUTTON_PAD) * SCREENS.size());
|
|
|
|
|
|
|
|
// windows
|
2023-09-06 20:36:48 +02:00
|
|
|
const auto WINDOWS_SCROLL_AREA_CONTENTS =
|
|
|
|
(QWidget*)TAB1->findChild<QWidget*>("windows")->findChild<QScrollArea*>("scrollArea_2")->findChild<QWidget*>("scrollAreaWidgetContents_2");
|
2022-12-03 22:27:49 +01:00
|
|
|
|
|
|
|
// loop over them
|
|
|
|
int windowIterator = 0;
|
2022-12-12 15:51:37 +01:00
|
|
|
for (auto& window : WINDOWLIST) {
|
2023-09-06 20:36:48 +02:00
|
|
|
QString text = QString::fromStdString(window.clazz + ": " + window.name);
|
2022-12-03 22:27:49 +01:00
|
|
|
|
|
|
|
QPushButton* button = new QPushButton(text, (QWidget*)WINDOWS_SCROLL_AREA_CONTENTS);
|
|
|
|
button->move(9, 5 + (BUTTON_HEIGHT + BUTTON_PAD) * windowIterator);
|
|
|
|
button->resize(BUTTON_WIDTH, BUTTON_HEIGHT);
|
|
|
|
|
2022-12-12 15:51:37 +01:00
|
|
|
mainPickerPtr->windowIDs[button] = window.id;
|
|
|
|
|
2023-02-03 19:41:28 +01:00
|
|
|
QObject::connect(button, &QPushButton::clicked, [=]() {
|
2023-09-06 20:36:48 +02:00
|
|
|
std::cout << (ALLOWTOKENBUTTON->isChecked() ? "r" : "");
|
|
|
|
std::cout << "/";
|
|
|
|
|
2022-12-12 15:51:37 +01:00
|
|
|
std::cout << "window:" << mainPickerPtr->windowIDs[button] << "\n";
|
2022-12-03 22:27:49 +01:00
|
|
|
pickerPtr->quit();
|
2022-12-04 22:12:57 +01:00
|
|
|
return 0;
|
2022-12-03 22:27:49 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
windowIterator++;
|
|
|
|
}
|
|
|
|
|
|
|
|
WINDOWS_SCROLL_AREA_CONTENTS->resize(WINDOWS_SCROLL_AREA_CONTENTS->size().width(), 5 + (BUTTON_HEIGHT + BUTTON_PAD) * windowIterator);
|
|
|
|
|
|
|
|
// lastly, region
|
2023-09-06 20:36:48 +02:00
|
|
|
const auto REGION_OBJECT = (QWidget*)TAB1->findChild<QWidget*>("region");
|
2022-12-03 22:27:49 +01:00
|
|
|
|
2023-09-06 20:36:48 +02:00
|
|
|
QString text = "Select region...";
|
2022-12-03 22:27:49 +01:00
|
|
|
|
|
|
|
QPushButton* button = new QPushButton(text, (QWidget*)REGION_OBJECT);
|
|
|
|
button->move(79, 80);
|
|
|
|
button->resize(321, 41);
|
2023-02-03 19:41:28 +01:00
|
|
|
QObject::connect(button, &QPushButton::clicked, [=]() {
|
2022-12-03 22:27:49 +01:00
|
|
|
auto REGION = execAndGet("slurp -f \"%o %x %y %w %h\"");
|
2023-09-06 20:36:48 +02:00
|
|
|
REGION = REGION.substr(0, REGION.length());
|
2022-12-03 22:27:49 +01:00
|
|
|
|
|
|
|
// now, get the screen
|
|
|
|
QScreen* pScreen = nullptr;
|
|
|
|
if (REGION.find_first_of(' ') == std::string::npos) {
|
|
|
|
std::cout << "error1\n";
|
|
|
|
pickerPtr->quit();
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
const auto SCREEN_NAME = REGION.substr(0, REGION.find_first_of(' '));
|
|
|
|
|
|
|
|
for (auto& screen : SCREENS) {
|
|
|
|
if (screen->name().toStdString() == SCREEN_NAME) {
|
|
|
|
pScreen = screen;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!pScreen) {
|
|
|
|
std::cout << "error2\n";
|
|
|
|
pickerPtr->quit();
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// get all the coords
|
|
|
|
try {
|
2023-09-06 20:36:48 +02:00
|
|
|
REGION = REGION.substr(REGION.find_first_of(' ') + 1);
|
2022-12-03 22:27:49 +01:00
|
|
|
const auto X = std::stoi(REGION.substr(0, REGION.find_first_of(' ')));
|
2023-09-06 20:36:48 +02:00
|
|
|
REGION = REGION.substr(REGION.find_first_of(' ') + 1);
|
2022-12-03 22:27:49 +01:00
|
|
|
const auto Y = std::stoi(REGION.substr(0, REGION.find_first_of(' ')));
|
2023-09-06 20:36:48 +02:00
|
|
|
REGION = REGION.substr(REGION.find_first_of(' ') + 1);
|
2022-12-03 22:27:49 +01:00
|
|
|
const auto W = std::stoi(REGION.substr(0, REGION.find_first_of(' ')));
|
2023-09-06 20:36:48 +02:00
|
|
|
REGION = REGION.substr(REGION.find_first_of(' ') + 1);
|
2022-12-03 22:27:49 +01:00
|
|
|
const auto H = std::stoi(REGION);
|
|
|
|
|
2023-09-06 20:36:48 +02:00
|
|
|
std::cout << (ALLOWTOKENBUTTON->isChecked() ? "r" : "");
|
|
|
|
std::cout << "/";
|
|
|
|
|
2022-12-03 22:27:49 +01:00
|
|
|
std::cout << "region:" << SCREEN_NAME << "@" << X - pScreen->geometry().x() << "," << Y - pScreen->geometry().y() << "," << W << "," << H << "\n";
|
|
|
|
pickerPtr->quit();
|
|
|
|
return 0;
|
|
|
|
} catch (...) {
|
|
|
|
std::cout << "error3\n";
|
|
|
|
pickerPtr->quit();
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::cout << "error4\n";
|
|
|
|
pickerPtr->quit();
|
|
|
|
return 1;
|
|
|
|
});
|
|
|
|
|
|
|
|
w.show();
|
|
|
|
return picker.exec();
|
|
|
|
}
|