grig_p
grig_pҚаз. 9, 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 Т.Ж.
  • (өңделген)

Добрый день!
Я сейчас мало работаю с 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 Т.Ж.

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

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

                Пікірлер

                Тек рұқсаты бар пайдаланушылар ғана пікір қалдыра алады.
                Кіріңіз немесе Тіркеліңіз
                Г

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

                • Нәтиже:66ұпай,
                • Бағалау ұпайлары-1
                t

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

                • Нәтиже:33ұпай,
                • Бағалау ұпайлары-10
                t

                Qt - Тест 001. Сигналы и слоты

                • Нәтиже:52ұпай,
                • Бағалау ұпайлары-4
                Соңғы пікірлер
                G
                GoattRockҚыр. 3, 2024, 1:50 Т.Қ.
                Linux жүйесінде файлдарды қалай көшіруге болады Задумывались когда-нибудь о том, как мы привыкли доверять свои вещи службам грузоперевозок? Сейчас такие услуги стали неотъемлемой частью нашей жизни, особенно когда речь идет о переездах между …
                d
                dblas5Шілде 5, 2024, 11:02 Т.Ж.
                QML - Сабақ 016. SQLite деректер қоры және онымен QML Qt-та жұмыс істеу Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                k
                kmssrАқп. 8, 2024, 6:43 Т.Қ.
                Qt Linux - Сабақ 001. Linux астында Autorun Qt қолданбасы как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
                АК
                Анатолий КононенкоАқп. 5, 2024, 1:50 Т.Ж.
                Qt WinAPI - Сабақ 007. Qt ішінде ICMP Ping арқылы жұмыс істеу Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
                Енді форумда талқылаңыз
                Evgenii Legotckoi
                Evgenii LegotckoiМаусым 24, 2024, 3:11 Т.Қ.
                добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
                F
                FynjyШілде 22, 2024, 4:15 Т.Ж.
                при создании qml проекта Kits есть но недоступны для выбора Поставил Qt Creator 11.0.2. Qt 6.4.3 При создании проекта Qml не могу выбрать Kits, они все недоступны, хотя настроены и при создании обычного Qt Widget приложения их можно выбрать. В чем может …
                BlinCT
                BlinCTМаусым 25, 2024, 1 Т.Ж.
                Нарисовать кривую в qml Всем привет. Имеется Лист листов с тосками, точки получаны интерполяцией Лагранжа. Вопрос, как этими точками нарисовать кривую? ChartView отпадает сразу, в qt6.7 появился новый элемент…
                BlinCT
                BlinCTМамыр 5, 2024, 5:46 Т.Ж.
                Написать свой GraphsView Всем привет. В Qt есть давольно старый обьект дял работы с графиками ChartsView и есть в 6.7 новый но очень сырой и со слабым функционалом GraphsView. По этой причине я хочу написать х…
                Evgenii Legotckoi
                Evgenii LegotckoiМамыр 2, 2024, 2:07 Т.Қ.
                Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Добрый день. По моему мнению - да, но то, что будет касаться вызовов к функционалу Андроида, может создать огромные трудности.

                Бізді әлеуметтік желілерде бақылаңыз