Использование SetWindowPos в С# для перемещения окон вокруг

У меня есть код ниже:

namespace WindowMover
{
    using System.Windows.Forms;

    static class Logic
    {
        [DllImport("user32.dll", EntryPoint = "SetWindowPos")]
        public static extern IntPtr SetWindowPos(IntPtr hWnd, int hWndInsertAfter, int x, int Y, int cx, int cy, int wFlags);

        public static void Move()
        {
            const short SWP_NOMOVE = 0X2;
            const short SWP_NOSIZE = 1;
            const short SWP_NOZORDER = 0X4;
            const int SWP_SHOWWINDOW = 0x0040;

            Process[] processes = Process.GetProcesses(".");
            foreach (var process in processes)
            {
                var handle = process.MainWindowHandle;
                var form = Control.FromHandle(handle);

                if (form == null) continue;

                SetWindowPos(handle, 0, 0, 0, form.Bounds.Width, form.Bounds.Height, SWP_NOZORDER | SWP_SHOWWINDOW);
            }
        }
    }
}

Предполагается, что каждое окно на моем рабочем столе перемещается на 0,0 (x, y) и сохраняет одинаковые размеры. Моя проблема в том, что только вызывающее приложение (построенное на С#) перемещается.

Должен ли я использовать что-то другое, кроме Control.FromHandle(IntPtr)? Будет ли это обнаружено только управление dotnet? Если да, то что я должен использовать?

Кроме того, второй 0 в SetWindowPos был просто случайным int я там, я не уверен, что использовать для int hWndInsertAfter

Как насчет процессов с несколькими окнами, например pidgin?

Ответ 1

Просто выньте свой Control.FromHandle и форму == null check. Вы должны иметь возможность просто:

IntPtr handle = process.MainWindowHandle;
if (handle != IntPtr.Zero)
{
    SetWindowPos(handle, 0, 0, 0, 0, 0, SWP_NOZORDER | SWP_NOSIZE | SWP_SHOWWINDOW);
}

Если вы добавите SWP_NOSIZE, он не изменит размер окна, но все равно изменит его.

Если вы хотите использовать все окна, а не только главное окно каждого процесса, вы можете рассмотреть возможность использования P/Invoke with EnumWindows вместо повторения в списке процессов и использования MainWindowHandle.

Ответ 2

Играется с этим. Посмотрите, помогает ли это.


using System.Windows.Forms;
using System.Runtime.InteropServices;
using System.Diagnostics;


namespace ConsoleTestApp
{
 class Program
 {
    [DllImport("user32.dll")]
    [return: MarshalAs(UnmanagedType.Bool)]
    static extern bool SetForegroundWindow(IntPtr hWnd);

    static void Main(string[] args)
    {

        Process[] processes = Process.GetProcesses();

        foreach (var process in processes)
        {
            Console.WriteLine("Process Name: {0} ", process.ProcessName); 

            if (process.ProcessName == "WINWORD")
            {
                IntPtr handle = process.MainWindowHandle;

                bool topMost =  SetForegroundWindow(handle); 
            }
        }
 }
}