Получение изображения, перетаскиваемого с веб-страницы в окно WPF - PullRequest
13 голосов
/ 09 декабря 2011

Я хочу, чтобы мое приложение WPF было целью перетаскивания, и я хочу иметь возможность перетаскивать изображение с любой веб-страницы.

Когда изображение перетаскивается с веб-страницы, оно, очевидно,Формат "DragImageBits", который можно десериализовать для типа ShDragImage .(Смотрите в нижней части вопроса, как я это определил)

Как мне преобразовать это в изображение WPF?

Вот моя текущая попытка.(Если кто-нибудь знает правильный способ сделать десириализацию, я весь слух)

   private void UserControl_Drop(object sender, System.Windows.DragEventArgs e)
    {

            string[] formats = data.GetFormats();

            // DragImageBits
            if (formats.Contains("DragImageBits"))
            {
            MemoryStream imageStream = data.GetData("DragImageBits") as MemoryStream;

            // Now I'm deserializing this, the only way I know how
            imageStream.Seek(0, SeekOrigin.Begin);
            BinaryReader br = new BinaryReader(imageStream);

            ShDragImage shDragImage;
            shDragImage.sizeDragImage.cx = br.ReadInt32();
            shDragImage.sizeDragImage.cy = br.ReadInt32();
            shDragImage.ptOffset.x = br.ReadInt32();
            shDragImage.ptOffset.y = br.ReadInt32();
            shDragImage.hbmpDragImage = new IntPtr(br.ReadInt32());
            shDragImage.crColorKey = br.ReadInt32();


            var systemDrawingBitmap = System.Drawing.Bitmap.FromHbitmap(shDragImage.hbmpDragImage);

В этот момент я получаю исключение типа System.Runtime.InteropServices.ExternalException, с сообщением, просто являющимся Generic GDI+ error.

Кто-нибудь знает, что я должен делать?

И вот определения поддерживающего класса.Я скопировал их из этой записи в блоге .

[System.Runtime.InteropServices.StructLayout(System.Runtime.InteropServices.LayoutKind.Sequential)]
    public struct Win32Point
    {
        public int x;
        public int y;
    }

    [System.Runtime.InteropServices.StructLayout(System.Runtime.InteropServices.LayoutKind.Sequential)]
    public struct Win32Size
    {
        public int cx;
        public int cy;
    }

    [System.Runtime.InteropServices.StructLayout(System.Runtime.InteropServices.LayoutKind.Sequential)]
    public struct ShDragImage
    {
        public Win32Size sizeDragImage;
        public Win32Point ptOffset;
        public IntPtr hbmpDragImage;
        public int crColorKey;
    }

1 Ответ

18 голосов
/ 12 декабря 2011

Вот что я узнал:

«DragImageBits» предоставляется оболочкой Windows и предназначен только для курсора перетаскивания, а не для конечных данных.Оболочка преобразует изображение в соответствующий курсор перетаскивания с помощью изменения размера и прозрачности.

Например, если перетащить это изображение:

Fully Opaque Image

SHDRAGIMAGE будет отображаться какэто:

enter image description here

Если вы действительно хотите извлечь изображение из SHDRAGIMAGE, вот код.(Частично снято с этого ответа )

        MemoryStream imageStream = data.GetData("DragImageBits") as MemoryStream;
        imageStream.Seek(0, SeekOrigin.Begin);
        BinaryReader br = new BinaryReader(imageStream);
        ShDragImage shDragImage;
        shDragImage.sizeDragImage.cx = br.ReadInt32();
        shDragImage.sizeDragImage.cy = br.ReadInt32();
        shDragImage.ptOffset.x = br.ReadInt32();
        shDragImage.ptOffset.y = br.ReadInt32();
        shDragImage.hbmpDragImage = new IntPtr(br.ReadInt32()); // I do not know what this is for!
        shDragImage.crColorKey = br.ReadInt32();
        int stride = shDragImage.sizeDragImage.cx * 4;
        var imageData = new byte[stride * shDragImage.sizeDragImage.cy];
        // We must read the image data as a loop, so it's in a flipped format
        for (int i = (shDragImage.sizeDragImage.cy - 1) * stride; i >= 0; i -= stride) 
        {
            br.Read(imageData, i, stride);
        }
        var bitmapSource = BitmapSource.Create(shDragImage.sizeDragImage.cx, shDragImage.sizeDragImage.cy,
                                                    96, 96,
                                                    PixelFormats.Bgra32,
                                                    null,
                                                    imageData,
                                                    stride);

Если вы хотите использовать DragImageBits по прямому назначению (в качестве перетаскиваемого изображения), см. этот блог дляпростой загружаемый пример.Таким образом, «DragImageBits» в значительной степени отвлекает от реальной проблемы, которая заключается в принятии изображения, перетаскиваемого с веб-страницы.

Перетаскивание изображения с веб-страницы становится сложным, поскольку Firefox, Chrome иIE9 все дает вам другой набор форматов.Кроме того, вы хотите обрабатывать как гиперссылку на изображение, так и гиперссылку на изображение, и они снова обрабатываются по-разному.

Google и Firefox предоставляют формат «text / html», который дает вам один элемент HTML в качестве изображения.Google выдает его в виде строки ASCII, а Firefox - в виде строки Unicode.Вот код, который я написал для его обработки:

     System.Windows.IDataObject data = e.Data;
        string[] formats = data.GetFormats();


        if (formats.Contains("text/html"))
        {

            var obj = data.GetData("text/html");
            string html = string.Empty;
            if (obj is string)
            {
                html = (string)obj;
            }
            else if (obj is MemoryStream)
            {
                MemoryStream ms = (MemoryStream)obj;
                byte[] buffer = new byte[ms.Length];
                ms.Read(buffer, 0, (int)ms.Length);
                if (buffer[1] == (byte)0)  // Detecting unicode
                {
                    html = System.Text.Encoding.Unicode.GetString(buffer);
                }
                else
                {
                    html = System.Text.Encoding.ASCII.GetString(buffer);
                }
            }
            // Using a regex to parse HTML, but JUST FOR THIS EXAMPLE :-)
            var match = new Regex(@"<img[^/]src=""([^""]*)""").Match(html);
            if (match.Success)
            {
                Uri uri = new Uri(match.Groups[1].Value);
                SetImageFromUri(uri);
            }
        }

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

И мою функцию SetImageFromUri:

    private void SetImageFromUri(Uri uri)
    {
        string fileName = System.IO.Path.GetTempFileName();
        using (WebClient webClient = new WebClient())
        {
            webClient.DownloadFile(uri, fileName);
        }
        using (FileStream fs = File.OpenRead(fileName))
        {
            byte[] imageData = new byte[fs.Length];
            fs.Read(imageData, 0, (int)fs.Length);
            this.ImageBinary = imageData;
        }
        File.Delete(fileName);
    }

Для IE9 вы можете работать с форматом FileDrop.Это хорошо работает в IE9.Chrome не поддерживает это.Firefox поддерживает его, но преобразует изображение в растровое изображение и преобразует прозрачные пиксели в черный.По этой причине формат «FileDrop» следует обрабатывать только в том случае, если формат «text.html» недоступен.

    else if (formats.Contains("FileDrop"))
    {
        var filePaths = (string[])data.GetData("FileDrop");
        using (var fileStream = File.OpenRead(filePaths[0]))
        {
            var buffer = new byte[fileStream.Length];
            fileStream.Read(buffer, 0, (int)fileStream.Length);
            this.ImageBinary = buffer;
        }
    }

Формат «FileDrop» не предоставляется, если перетащить гиперссылку изображения изIE9.Я не понял, как перетащить изображение из гиперссылки изображения в IE9 на свой элемент управления изображением.

Дополнительная информация

Если вы используете этот пример, но все еще должны преобразовать эти двоичные данные в изображение, вот полезный фрагмент кода:

                BitmapImage sourceImage = new BitmapImage();
                using (MemoryStream ms = new MemoryStream(imageBinary))
                {
                    sourceImage.BeginInit();
                    sourceImage.CacheOption = BitmapCacheOption.OnLoad;
                    sourceImage.StreamSource = ms;
                    sourceImage.EndInit();
                }
...