© EVILEG 2015-2018
Рекомендует хостинг
TIMEWEB
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/

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

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

Ответы

Только авторизованные пользователи могут отвечать на форуме.
Пожалуйста, Авторизуйтесь или Зарегистрируйтесь
15 июня 2018 г. 12:42
Nicky

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

  • Результат 100 баллов
  • Очки рейтинга 10
15 июня 2018 г. 12:36
Nicky

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

  • Результат 57 баллов
  • Очки рейтинга -2
15 июня 2018 г. 12:29
Nicky

C++ - Тест 001. Первая программа и типы данных

  • Результат 46 баллов
  • Очки рейтинга -6
Последние комментарии
18 июня 2018 г. 7:12
EVILEG

PyQt5 - Урок 007. Работаем с QML QtQuick (Сигналы и слоты)

Я вот сейчас банальность скажу, но у меня всё работало. Так что даже и не знаю, надо на код смотреть, что ещё у вас добавлено или отсутствует из библиотек. P/S/ Извините, вы сейчас вс...
18 июня 2018 г. 7:10
EVILEG

Qt/C++ - Урок 042. PopUp уведомление в стиле Gnome с помощью Qt

Недоработки, вряд ли этот зверь вообще является официально поддерживаемым
18 июня 2018 г. 7:01
EVILEG

QML - Урок 016. База данных SQLite и работа с ней в QML Qt

что-то мне сдаётся, что здесь просто пересобрать проект нужно с удалением build каталога
18 июня 2018 г. 7:00
EVILEG

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

Если зарыться в API системы, то, думаю, что можно, тут тоже использовался WinAPI.
16 июня 2018 г. 15:19
pro100belik

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

А можно по ID процесса  выводить на передний план окно? myProcess->processId();
Сейчас обсуждают на форуме
19 июня 2018 г. 7:56
EVILEG

как редактировать порядок обхода этементов по нажатию TAB в Qt5 qml

Что-то наподобие такого TextField { Keys.onReturnPressed: nextItemInFocusChain().forceActiveFocus()}
19 июня 2018 г. 6:31
kabanov

Как сохранить фокус в TextField после перезагрузки модели

Rectangle { ListView { id: listView delegate: Item { id: cDelegate Item { Row { ComboBox { ...
18 июня 2018 г. 10:51
alex_lip

Qml and JavaScript

В том то и дело что просто в JS так нельзя Если использовать state - onReleased - не нужен вот так все работает Text { ...
18 июня 2018 г. 7:16
EVILEG

почему не выполняется код после вызова слота?

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

Рекомендуемые страницы