Могу ли я получить растровое изображение произвольного окна в другом процессе приложения?

Я пытаюсь автоматизировать стороннее приложение Win32, в котором я хочу записывать графическое содержимое определенного окна через определенные промежутки времени. Я нахожусь на ранних этапах этого, и в настоящее время я пытаюсь использовать Microsoft UI Automation через C#, чтобы выполнять большую часть взаимодействия между моим клиентским приложением и внешним приложением. Теперь я могу заставить внешнее приложение делать то, что мне нужно, но теперь я хочу захватить графику из определенного окна, которое кажется каким-то сторонним элементом управления, нарисованным владельцем. Как я могу это сделать? Окно, которое я хочу захватить, отмечено красным прямоугольником на этом изображении:

Мне нужно то, что в красном прямоугольнике

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

var p = Process.Start("c:\myapp.exe");
var mainForm = AutomationElement.FromHandle(p.MainWindowHandle);
// "workspace" below is the window whose content I want to capture.
var workspace = mainForm.FindFirst(TreeScope.Descendents,
                    new PropertyCondition(AutomationElement.ClassNameProperty, "AfxFrameOrView70u"));
var rect = (Rect) workspace.GetCurrentPropertyValue(AutomationElement.BoundingRectangleProperty);
using (var bmp = new Bitmap((int)rect.Width, (int)rect.Height))
{
    using (var g = Graphics.FromImage(bmp))
    {
        g.CopyFromScreen((int)rect.Left, (int)rect.Top, 0, 0, new Size((int)rect.Width, (int)rect.Height));
        bmp.Save(@"c:\screenshot.png", ImageFormat.Png);
    }
}

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

Я прочитал несколько предложений по отправке сообщения WM_PRINT в окно. Этот вопрос/ответ несколько месяцев назад казался многообещающе, но когда я использую этот код, я просто получаю белый прямоугольник без фактического содержимого моего элемента управления.

var prop = (int)workspace.GetCurrentPropertyValue(AutomationElement.NativeWindowHandleProperty);
var hwnd = new IntPtr(prop);
using ( var bmp2 = new Bitmap((int)rect.Width, (int)rect.Height))
{
    using (Graphics g = Graphics.FromImage(bmp2))
    {
        g.FillRectangle(SystemBrushes.Control, 0, 0, (int)rect.Width, (int)rect.Height);
        try
        {
            SendMessage(hwnd, WM_PRINT, g.GetHdc().ToInt32(), (int)(DrawingOptions.PRF_CHILDREN | DrawingOptions.PRF_CLIENT | DrawingOptions.PRF_OWNED));
        }
        finally
        {
            g.ReleaseHdc();
        }
        bmp2.Save(@"c:\screenshot.bmp");
    }
}

Итак, во-первых, возможно ли мне надежно сохранить растровое изображение содержимого окна? Если да, то как лучше всего и что не так с моей попыткой WM_PRINT с SendMessage?


person Chris Farmer    schedule 14.05.2010    source источник


Ответы (2)


Эта модификация PrintWindow примера API на сайте pinvoke.net кажется сделали трюк.

Bitmap bmp = new Bitmap((int)rect.Width, (int)rect.Height);
Graphics memoryGraphics = Graphics.FromImage(bmp);
IntPtr dc = memoryGraphics.GetHdc();
bool success = PrintWindow(hwnd, dc, 0);
memoryGraphics.ReleaseHdc(dc);
bmp.Save(@"c:\screenshot.bmp");

Это работает, если приложение закрыто другим окном, но не работает, если приложение свернуто. Я думаю, я могу жить с этим.

person Chris Farmer    schedule 14.05.2010

Не существует надежного способа получить растровое изображение из другого приложения, если это приложение не находится сверху. Это связано с тем, что элементы управления приложения даже не отображаются, если приложение не видимо, и Windows даже не обязательно запоминает, каким было последнее известное содержимое элемента управления после того, как этот элемент управления теряет самую верхнюю позицию в z-порядке.

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

person JSBձոգչ    schedule 14.05.2010