使用pictureBox的缩放模式裁剪位图克隆

3
我在互联网上找到了一个裁剪图像的函数,如果pictureBox源的sizeMode是正常的,则它可以很好地完成工作。但是当pictureBox sizeMode为缩放时,它仍然像正常sizeMode一样被克隆。
如何克隆与缩放的pictureBox相同?而不是原始正常位图的大小?
public static Bitmap CropBitmap(Bitmap bitmap, int x, int y, int w, int h)
{
    Rectangle rect = new Rectangle(x, y, w, h);
    Bitmap cropped = bitmap.Clone(rect, bitmap.PixelFormat);
    return cropped;
}

并且可以像这样使用

pictureBox2.Image = CropBitmap((Bitmap)pictureBox1.Image.Clone(), 35, 0, 110, 150);

从技术上讲,您并没有裁剪“Bitmap”对象,而是图片框的输出,除了“SizeMode”为Normal或AutoSize之外,其他情况都不同。 - Ivan Stoev
1个回答

7
最简单的方法是使用DrawToBitmap方法从图片框中获取输出,无论SizeMode如何,然后像这样裁剪该输出:
public static Bitmap Crop(PictureBox pb, int x, int y, int w, int h)
{
    var rect = pb.ClientRectangle;
    using (var output = new Bitmap(rect.Width, rect.Height, pb.Image.PixelFormat))
    {
        pb.DrawToBitmap(output, rect);
        return output.Clone(new Rectangle(x, y, w, h), output.PixelFormat);
    }
}

然而,以上方法的缺点是它会真正裁剪可能被缩放的图像。
如果您确实想要裁剪原始图像,您需要将传递的矩形(我假设它在图片框客户端坐标系中)映射到原始图像坐标系中。
如果图片框提供了类似于ClientToScreen的方法ClientToImage就好了,但它没有,所以我们需要从Reference Source中提取SizeMode逻辑。
新方法如下:
public static class ImageUtils
{
    public static Bitmap CropImage(this PictureBox pb, int x, int y, int w, int h)
    {
        var imageRect = pb.GetImageRectangle();
        var image = pb.Image;
        float scaleX = (float)image.Width / imageRect.Width;
        float scaleY = (float)image.Height / imageRect.Height;
        var cropRect = new Rectangle();
        cropRect.X = Scale(x - imageRect.X, scaleX, image.Width);
        cropRect.Y = Scale(y - imageRect.Y, scaleY, image.Height);
        cropRect.Width = Scale(w, scaleX, image.Width - cropRect.X);
        cropRect.Height = Scale(h, scaleY, image.Height - cropRect.Y);
        var result = new Bitmap(cropRect.Width, cropRect.Height, image.PixelFormat);
        using (var g = Graphics.FromImage(result))
            g.DrawImage(image, new Rectangle(new Point(0, 0), cropRect.Size), cropRect, GraphicsUnit.Pixel);
        return result;
    }

    static int Scale(int value, float scale, int maxValue)
    {
        int result = (int)(value * scale);
        return result < 0 ? 0 : result > maxValue ? maxValue : result;
    }

    public static Rectangle GetImageRectangle(this PictureBox pb)
    {
        var rect = pb.ClientRectangle;
        var padding = pb.Padding;
        rect.X += padding.Left;
        rect.Y += padding.Top;
        rect.Width -= padding.Horizontal;
        rect.Height -= padding.Vertical;
        var image = pb.Image;
        var sizeMode = pb.SizeMode;
        if (sizeMode == PictureBoxSizeMode.Normal || sizeMode == PictureBoxSizeMode.AutoSize)
            rect.Size = image.Size;
        else if (sizeMode == PictureBoxSizeMode.CenterImage)
        {
            rect.X += (rect.Width - image.Width) / 2;
            rect.Y += (rect.Height - image.Height) / 2;
            rect.Size = image.Size;
        }
        else if (sizeMode == PictureBoxSizeMode.Zoom)
        {
            var imageSize = image.Size;
            var zoomSize = pb.ClientSize;
            float ratio = Math.Min((float)zoomSize.Width / imageSize.Width, (float)zoomSize.Height / imageSize.Height);
            rect.Width = (int)(imageSize.Width * ratio);
            rect.Height = (int)(imageSize.Height * ratio);
            rect.X = (pb.ClientRectangle.Width - rect.Width) / 2;
            rect.Y = (pb.ClientRectangle.Height - rect.Height) / 2;
        }
        return rect;
    }
}

我一直在寻找一种方法,可以裁剪图片框的“缩放”输出,你的DrawToBitmap示例对我帮助很大,谢谢! - daddywoodland

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接