Slice Large WPF Canvas Into Many Small PNG Images

时光怂恿深爱的人放手 提交于 2019-12-14 01:23:07

问题


Assume I have very large (3000 X 4000) WPF canvas.

I'm looking for the best way to slice this canvas into a bunch of n by n .png image tiles.

Any suggestions?


回答1:


Not sure about either best or easiest but here's a way :)

private void SaveFrameworkElement(FrameworkElement frameworkElement)
{
    BitmapImage bitmapImage = VisualToBitmapImage(frameworkElement);
    for (int startX = 0; startX <= 400; startX += 100)
    {
        for (int startY = 0; startY <= 100; startY += 100)
        {
            SaveImage(bitmapImage, startX, startY, 100, 100, "C:\\CutImage_" + startX.ToString() + "-" + startY.ToString() + ".png");
        }
    }
}

public BitmapImage VisualToBitmapImage(FrameworkElement frameworkElement)
{
    RenderTargetBitmap rtb = new RenderTargetBitmap((int)frameworkElement.ActualWidth,
                                                    (int)frameworkElement.ActualHeight,
                                                    96d,
                                                    96d,
                                                    PixelFormats.Default);
    rtb.Render(frameworkElement);

    MemoryStream stream = new MemoryStream();
    PngBitmapEncoder encoder = new PngBitmapEncoder();
    encoder.Frames.Add(BitmapFrame.Create(rtb));
    encoder.Save(stream);

    BitmapImage bitmapImage = new BitmapImage();
    bitmapImage.BeginInit();
    bitmapImage.StreamSource = stream;
    bitmapImage.EndInit();

    return bitmapImage;            
}
public void SaveImage(BitmapImage sourceImage,
                      int startX,
                      int startY,
                      int width,
                      int height,
                      string filePath)
{
    TransformGroup transformGroup = new TransformGroup();
    TranslateTransform translateTransform = new TranslateTransform();
    translateTransform.X = -startX;
    translateTransform.Y = -startY;
    transformGroup.Children.Add(translateTransform);

    DrawingVisual vis = new DrawingVisual();
    DrawingContext cont = vis.RenderOpen();
    cont.PushTransform(transformGroup);
    cont.DrawImage(sourceImage, new Rect(new Size(sourceImage.PixelWidth, sourceImage.PixelHeight)));
    cont.Close();

    RenderTargetBitmap rtb = new RenderTargetBitmap(width, height, 96d, 96d, PixelFormats.Default);
    rtb.Render(vis);

    FileStream stream = new FileStream(filePath, FileMode.Create);
    PngBitmapEncoder encoder = new PngBitmapEncoder();
    encoder.Frames.Add(BitmapFrame.Create(rtb));
    encoder.Save(stream);
    stream.Close();
}



回答2:


You can do something like this (not my code, taken from msdn social):

public static RenderTargetBitmap GetJpgImage(UIElement targetUIElement, double scale, int quality)
{
double actualHeight = targetUIElement.RenderSize.Height;
double actualWidth = targetUIElement.RenderSize.Width;

double renderedHeight = actualHeight * scale;
double renderedWidth = actualWidth * scale;

RenderTargetBitmap renderTarget = new RenderTargetBitmap((int)renderedWidth, (int)renderedHeight, 96, 96, PixelFormats.Pbgra32);
VisualBrush sourceBrush = new VisualBrush(targetUIElement);

DrawingVisual drawingVisual = new DrawingVisual();
DrawingContext drawingContext = drawingVisual.RenderOpen();

using (drawingContext)
{
    drawingContext.PushTransform(new ScaleTransform(scale, scale));
    drawingContext.DrawRectangle(sourceBrush, null, new Rect(new System.Windows.Point(0, 0), new System.Windows.Point(actualWidth, actualHeight)));
}
renderTarget.Render(drawingVisual);

JpegBitmapEncoder jpgEncoder = new JpegBitmapEncoder();
jpgEncoder.QualityLevel = quality;
jpgEncoder.Frames.Add(BitmapFrame.Create(renderTarget));
MemoryStream memoryStream = new MemoryStream();
jpgEncoder.Save(memoryStream);

File.WriteAllBytes(@"D:\a.jpg", memoryStream.ToArray());

return renderTarget;
}

Elaborate on that code to section the JPG off into squares, then you can use a Grid, programmatically creating a bunch of columns and rows, and using the Grid.Row and Grid.Column attached properties on the images programmatically to specify which (sliced) images are present.



来源:https://stackoverflow.com/questions/4453059/slice-large-wpf-canvas-into-many-small-png-images

标签
易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!