9 октября 2017 г. 8:37

Qt. Показать запущенное приложение поверх своего.

Здравствуйте!
Есть у меня несколько приложений. Часть на Qt, часть - нет. И есть главное приложение, которое должно вызывать эти несколько приложений. Причем, если приложение не запущено, то оно должно запускаться, а если запущено, его окно должно выводиться на передний план. Для вызова приложений использую QProcess.
Выглядит это следующим образом.
Создание процесса:

QProcess *app;
bool pw = false;
...
void createProcess()
{
    app = new QProcess(this);
    app->setProgram("MyProgram.exe");
    app->setArguments(QStringList() << "arg1" << "arg2");
    connect(app, SIGNAL(started()), this, SLOT(onProcStarted()));
    connect(app, SIGNAL(finished()), this, SLOT(onProcFinished()));
}
Вызов:
if (!pw){
    app->start(QIODevice::ReadOnly);
} else {
    HWND hWnd = FindWindow(nullptr, L"Заголовок главного окна");
    if (hWnd > 0){
        ShowWindow(hWnd, SW_RESTORE);
        SetForegroundWindow(hWnd);
    }
}
Слоты:
void onProcStarted()
{
    pw = true;
}

void onProcFinished(int)
{
    pw = false;
}
Как я понял, QProcess не умеет выводить на передний план окно запущенного процесса. Поэтому приходится прибегать к WinApi. Но показ окна не работает. Причем, не работает функция FindWindow. Она не возвращает корректный хэндл. Хотя окно есть.
Я написал небольшую консольную программу на Delphi, используя эти же вызовы. Она работает и показывает нужное окно. Вот ее код:
program BTFArmAdmin;

{$APPTYPE CONSOLE}

uses
  Windows, SysUtils;

const
  WinCaption = 'Заголовок окна';

var
  hWndArm : hwnd;


begin
  try
    hWndArm := FindWindow(nil, WinCaption);
    if hWndArm > 0 then
    begin
      ShowWindow(hWndArm, SW_RESTORE);
      SetForegroundWindow(hWndArm);
    end;

  except
    on E:Exception do
      Writeln(E.Classname, ': ', E.Message);
  end;
end.
Вот эту программу я вызываю из главной программы Qt для показа окна.
Но это некрасиво.
Как сделать так, чтобы это работало из Qt? Что не так у меня в коде?
Заранее очень благодарен за ответ!

Пробовал делать так, но все равно не работает:
Q_PID pi = app->pid();
ShowWindow((HWND)pi->hProcess, SW_RESTORE);
SetForegroundWindow((HWND)pi->hProcess);
  • #
  • отредактировано 9 октября 2017 г. 9:07
  • 9 октября 2017 г. 9:00

Добрый день!
Я сейчас мало работаю с WinAPI , но могу предположить, что неработоспособность функции FindWindow заключается в кодировке. Скорее всего нужно передать массив символов в кодировке cp1251 , а там передаётся что-то другое.


Что касается QProcess , то он возвращает id процесса через метод QProcess::processId()
И кое-где на форумах можно найти варианты, когда перебирают все хендлеры окон, сравнивая текущий id процесса с тем, который нужен. Например, что-нибудь такое
void GetProcessMainWindows (DWORD dwProcessID, vector <HWND> &vWindows)
{
    HWND hwnd = NULL;

    do 
    {
        hwnd = FindWindowEx (NULL, hwnd, NULL, NULL);
        DWORD dwPID = 0;
        GetWindowThreadProcessId (hwnd, &dwPID);
        if (dwPID == dwProcessID)
           vWindows.push_back (hwnd);
    }
    while (hwnd != NULL);
}
Смысл этого кода заключается в том, что пытаемся найти все окна вне зависимости от названия, дёрнуть processId каждого окна и проверить, соответствует ли он тому, что нам нужен, если да, то загружаем в вектор окон. (Просто окон может быть несколько для одного процесса).

А вообще, скорее всего неработоспособность вашего первого варианта заключается в кодировке текста, который передаётся в FindWindow
  • #
  • Ответ был помечен как решение
  • 9 октября 2017 г. 14:09

Спасибо большое!
С Вашей подсказкой получилось.
Вот код:

        QTextCodec *codec = QTextCodec::codecForName("UTF-8");
        QString s = codec->toUnicode("Заголовок главного окна");
        LPCWSTR lps = (LPCWSTR)s.utf16();

        HWND hWnd = FindWindow(nullptr, lps);
        if (hWnd > 0) {
            ShowWindow(hWnd, SW_RESTORE);
            SetForegroundWindow(hWnd);
        }

Пожалуйста.

С таким вопросом впору и заметку накидать. Интересный момент с кодировкой.
  • #
  • 11 октября 2017 г. 7:27

Спасибо.
А как это сделать?

  • grig_p
  • #
  • отредактировано 11 октября 2017 г. 7:35
  • 11 октября 2017 г. 7:34

Если есть желание, то можете оставлять заметки на сайте в виде статей, это может быть всего несколько абзацев о какой-нибудь полезняшке в программировании.


Например, была такая проблема -> Решил её таким образом. Или захотел сделать что-то -> сделал это так.

Справа вверху в меню есть пункт "Написать статью" .
Кликаете, пишете материал, нажимаете опубликовать или сохранить, если недописали и планируете вернуться поздней. Статья проходит модерацию, правку вёрстки и одобряется, будет видна пользователям сайта после этого. Получаете статус автора и статьи уже публикуются сразу с постмодерацией (иногда приходится править вёрстку)

Для отделения текста, который будет показан в ленте новостей от всего остального текста статьи используете горизонтальную линию. В редакторе статьей есть кнопка для добавления горизонтальной линии. Работает наподобие ката на Хабре.
Статьи можно написать на двух языках, там две вкладки, но это не обязательно (я всё равно потом добавляю перевод и слегка правлю вёрстку). То есть достаточно и русского варианта текста.
  • #
  • 31 октября 2017 г. 7:01

Написал.
Прошу прощения за задержку:
https://evileg.com/post/295/

Круто)) Спасибо

Немного подкорректировал и одобрил публикацию.
Если вдруг ещё будут полезные идеи, то ждёмс. Но один момент на заметку. Для отделения текста, который виден в ленте новостей от остального текста, который виден на странице статьи используйте "Вставку горизонтальной линии" Она выступает в качестве разделителя.
Спасибо ))))

Ответы

Только авторизованные пользователи могут отвечать на форуме.
Пожалуйста, Авторизуйтесь или Зарегистрируйтесь
25 февраля 2018 г. 14:09
exxtra_noise

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат 100 баллов
  • Очки рейтинга 10
25 февраля 2018 г. 14:06
exxtra_noise

C++ - Тест 004. Указатели, Массивы и Циклы

  • Результат 60 баллов
  • Очки рейтинга -1
25 февраля 2018 г. 10:39
exxtra_noise

C++ - Тест 003. Условия и циклы

  • Результат 100 баллов
  • Очки рейтинга 10
Последние комментарии
26 февраля 2018 г. 0:55
soz7557

Qt/C++ - Урок 029. Изображение в базе данных в Qt – Сохранение и Восстановление

thanks, but Id should be the same the one as i select the image in tree view.

25 февраля 2018 г. 21:53
Console

Выпуск Qt 5.10

Здравствуйте.Планируется ли урок по Qt Network Authorization? Всё же интересная тема и информации маловато в интернете.

25 февраля 2018 г. 19:54
EVILEG

GameDev на Qt - Урок 4. Обнаружение коллизий в Qt (2D)

Сначала нужно что-нибудь написать, прежде чем это оптимизировать. А вообще все оптимизации исходят из совокупности условий, а не из того, что есть один танк и N треугольников. Да и области вид...

25 февраля 2018 г. 19:31
romankoshelev

GameDev на Qt - Урок 4. Обнаружение коллизий в Qt (2D)

А как насчет оптимизации. Тут ведь например будет ездить N треугольников, для каждого проводить линию и смотреть пересечения? +Это же происходит много раз в секунду.

25 февраля 2018 г. 16:15
EVILEG

GameDev на Qt - Урок 4. Обнаружение коллизий в Qt (2D)

Как вариант, использовать QLineF. Условно взять линию от положения танка, до положения треугольника. И проверить пересечение этой линии с другими линиями на карте с помощью метода intersect. Этот м...

Сейчас обсуждают на форуме
25 февраля 2018 г. 11:31
Mic78

How to retrieve a QCandlestickSet item from a QChartView?

Sorry, in the last sentence I wanted to write: "For that I need to know the maximum value of the QCandlestickSets in the zoomed area."

25 февраля 2018 г. 10:02
EVILEG

Проблема с ComboBox

Да, в принципе идея понятна. Можно воспользоваться одной исходной моделью и делать по ней поиск. Найденные элементы добавлять в модель для отображения. При этом исходная модель буд...

25 февраля 2018 г. 9:07
EVILEG

Qt управление окнами других программ

Добрый день! Это всё делается через WinAPI. В Qt не занимаются разработкой настолько платформозависимого функционала, который нужен единицам. Не знаю, что там было в Delphi, возможн...

21 февраля 2018 г. 13:26
sol11

Qtableviev после сортировки

Спасибо, всё заработало :) Единственное вот тут row на id поменял и всё круто :)) if(id == -1){ model->insertRow(model->rowCount(QModelIndex())); map...

20 февраля 2018 г. 13:18
alex_lip

Разбить один qml файл на несколько составляющих

Да спасибо. Просто после необходимости специфичных названий для файла - стараюсь обращать внимание на любую мелочь.