grig_p
grig_p9 октября 2017 г. 4: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);
Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

Вам это нравится? Поделитесь в социальных сетях!

7
Evgenii Legotckoi
  • 9 октября 2017 г. 5: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
    grig_p
    • 9 октября 2017 г. 10: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);
            }
    
      Evgenii Legotckoi
      • 9 октября 2017 г. 10:25

      Пожалуйста.

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

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

          Evgenii Legotckoi
          • 11 октября 2017 г. 3:34
          • (ред.)

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


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

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

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

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

              Evgenii Legotckoi
              • 31 октября 2017 г. 4:35

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

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

                Комментарии

                Только авторизованные пользователи могут публиковать комментарии.
                Пожалуйста, авторизуйтесь или зарегистрируйтесь
                г
                • ги
                • 23 апреля 2024 г. 15:51

                C++ - Тест 005. Структуры и Классы

                • Результат:41баллов,
                • Очки рейтинга-8
                l
                • laei
                • 23 апреля 2024 г. 9:19

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

                • Результат:10баллов,
                • Очки рейтинга-10
                l
                • laei
                • 23 апреля 2024 г. 9:17

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

                • Результат:50баллов,
                • Очки рейтинга-4
                Последние комментарии
                k
                kmssr8 февраля 2024 г. 18:43
                Qt Linux - Урок 001. Автозапуск Qt приложения под Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
                АК
                Анатолий Кононенко5 февраля 2024 г. 1:50
                Qt WinAPI - Урок 007. Работаем с ICMP Ping в Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
                EVA
                EVA25 декабря 2023 г. 10:30
                Boost - статическая линковка в CMake проекте под Windows Ошибка LNK1104 часто возникает, когда компоновщик не может найти или открыть файл библиотеки. В вашем случае, это файл libboost_locale-vc142-mt-gd-x64-1_74.lib из библиотеки Boost для C+…
                J
                JonnyJo25 декабря 2023 г. 8:38
                Boost - статическая линковка в CMake проекте под Windows Сделал всё по-как у вас, но выдаёт ошибку [build] LINK : fatal error LNK1104: не удается открыть файл "libboost_locale-vc142-mt-gd-x64-1_74.lib" Хоть убей, не могу понять в чём дел…
                G
                Gvozdik18 декабря 2023 г. 21:01
                Qt/C++ - Урок 056. Подключение библиотеки Boost в Qt для компиляторов MinGW и MSVC Для решения твой проблемы добавь в файл .pro строчку "LIBS += -lws2_32" она решит проблему , лично мне помогло.
                Сейчас обсуждают на форуме
                G
                Gar22 апреля 2024 г. 5:46
                Clipboard Как скопировать окно целиком в clipb?
                DA
                Dr Gangil Academics20 апреля 2024 г. 7:45
                Unlock Your Aesthetic Potential: Explore MSC in Facial Aesthetics and Cosmetology in India Embark on a transformative journey with an msc in facial aesthetics and cosmetology in india . Delve into the intricate world of beauty and rejuvenation, guided by expert faculty and …
                a
                a_vlasov14 апреля 2024 г. 6:41
                Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Евгений, добрый день! Такой вопрос. Верно ли следующее утверждение: Любое Android-приложение, написанное на Java/Kotlin чисто теоретически (пусть и с большими трудностями) можно написать и на C+…
                Павел Дорофеев
                Павел Дорофеев14 апреля 2024 г. 2:35
                QTableWidget с 2 заголовками Вот тут есть кастомный QTableView с многорядностью проект поддерживается, обращайтесь
                f
                fastrex4 апреля 2024 г. 4:47
                Вернуть старое поведение QComboBox, не менять индекс при resetModel Добрый день! У нас много проектов в которых используется QComboBox, в версии 5.5.1, когда модель испускает сигнал resetModel, currentIndex не менялся. В версии 5.15 при resetModel происходит try…

                Следите за нами в социальных сетях