จะครอบตัดรูปภาพโดยใช้ C # ได้อย่างไร


คำตอบ:


228

คุณสามารถใช้Graphics.DrawImageเพื่อวาดภาพที่ถูกครอบตัดลงบนวัตถุกราฟิกจากบิตแมป

Rectangle cropRect = new Rectangle(...);
Bitmap src = Image.FromFile(fileName) as Bitmap;
Bitmap target = new Bitmap(cropRect.Width, cropRect.Height);

using(Graphics g = Graphics.FromImage(target))
{
   g.DrawImage(src, new Rectangle(0, 0, target.Width, target.Height), 
                    cropRect,                        
                    GraphicsUnit.Pixel);
}

3
แค่ทราบว่าลายเซ็นของ DrawImage () ไม่ถูกต้อง มันหายไปพารามิเตอร์ GraphicsUnit
นาธานเทย์เลอร์

2
อาร์กิวเมนต์ที่สองคือ rect เป้าหมายไม่ใช่ rect ครอบตัด
axk

8
วิธีนี้DrawImageUnscaledAndClippedมีประสิทธิภาพมากกว่าDrawImageการปลูกพืชหรือไม่?
Ivan Kochurkin

270

ลองใช้ลิงค์นี้: http://www.switchonthecode.com/tutorials/csharp-tutorial-image-editing-saving-cropping-and-resizing

private static Image cropImage(Image img, Rectangle cropArea)
{
   Bitmap bmpImage = new Bitmap(img);
   return bmpImage.Clone(cropArea, bmpImage.PixelFormat);
}

56
เห็นด้วย แต่โปรดทราบว่าถ้า cropArea ข้ามเขตแดน img จะทำให้เกิดข้อยกเว้น "หน่วยความจำไม่เพียงพอ"
ChrisJJ

1
@ KvanTTT ทั้งคู่ค่อนข้างช้าถ้าคุณต้องการครอบตัดภาพใหญ่ให้เล็กลง
JBeurer

1
@ChrisJJ คุณสามารถอธิบายเพิ่มเติมได้หรือไม่ หรือให้วิธีแก้ปัญหาสำหรับปัญหานั้น
raym0nd

1
@ raym0 และฉันเดาว่าวิธีแก้ปัญหาคือเพื่อให้แน่ใจว่าขนาดของสี่เหลี่ยมผืนผ้าของคุณนั้นไม่ใหญ่ไปกว่าภาพ
stuartdotnet

4
เว็บไซต์ของพวกเขาไม่ทำงาน ทุกคนได้รับรหัสจากเว็บไซต์หรือไม่
Sangam

55

ง่ายกว่าคำตอบที่ยอมรับคือ:

public static Bitmap cropAtRect(this Bitmap b, Rectangle r)
{
    using (Bitmap nb = new Bitmap(r.Width, r.Height))
    using (Graphics g = Graphics.FromImage(nb))
    {
        g.DrawImage(b, -r.X, -r.Y);
        return nb;
    }
}

และจะหลีกเลี่ยงความเสี่ยงยกเว้น " หน่วยความจำไม่เพียงพอ" ของคำตอบที่ง่ายที่สุด

โปรดทราบว่าBitmapและGraphicsเป็นIDisposableด้วยเหตุนี้usingคำสั่ง

แก้ไข : ฉันคิดว่าสิ่งนี้ใช้ได้กับ PNG ที่บันทึกโดยBitmap.Saveหรือ Paint.exe แต่ล้มเหลวด้วย PNG ที่บันทึกโดยเช่นPaint Shop Pro 6 - เนื้อหาถูกแทนที่ การเติมGraphicsUnit.Pixelจะให้ผลลัพธ์ที่ผิดไป บางที PNGs ที่ล้มเหลวเหล่านี้อาจมีข้อผิดพลาด


5
คำตอบที่ดีที่สุดที่นี่ควรได้รับคำตอบ ฉันกำลังประสบกับ 'หน่วยความจำไม่เพียงพอ' รวมถึงโซลูชันอื่น ๆ ครั้งนี้ทำงานครั้งแรก
c0d3p03t

ฉันไม่เข้าใจว่าทำไมการเพิ่ม GraphicsUnit.Pixel ให้ผลที่ไม่ถูกต้อง แต่มันไม่แน่นอน
DOKKA

รูปภาพของฉันถูกครอบตัดด้วยขนาดที่ถูกต้อง แต่ไม่ถูกต้อง X / Y จนกระทั่งฉันเรียก SetResolution บนรูปภาพเป้าหมายตามที่แนะนำในคำตอบโดย @IntellyDev
Brent Keller

7
คำตอบนี้ทำให้วัตถุ Grphics รั่ว
TaW

2
BitmapและGraphicsเป็นIDisposable- เพิ่มusingประโยค
เดฟ thieben

7

ใช้ bmp.SetResolution(image.HorizontalResolution, image .VerticalResolution);

สิ่งนี้อาจจำเป็นต้องทำแม้ว่าคุณจะใช้คำตอบที่ดีที่สุดที่นี่โดยเฉพาะอย่างยิ่งถ้าภาพของคุณดีมากและความละเอียดไม่ได้อยู่ที่ 96.0

ตัวอย่างการทดสอบของฉัน:

    static Bitmap LoadImage()
    {
        return (Bitmap)Bitmap.FromFile( @"e:\Tests\d_bigImage.bmp" ); // here is large image 9222x9222 pixels and 95.96 dpi resolutions
    }

    static void TestBigImagePartDrawing()
    {
        using( var absentRectangleImage = LoadImage() )
        {
            using( var currentTile = new Bitmap( 256, 256 ) )
            {
                currentTile.SetResolution(absentRectangleImage.HorizontalResolution, absentRectangleImage.VerticalResolution);

                using( var currentTileGraphics = Graphics.FromImage( currentTile ) )
                {
                    currentTileGraphics.Clear( Color.Black );
                    var absentRectangleArea = new Rectangle( 3, 8963, 256, 256 );
                    currentTileGraphics.DrawImage( absentRectangleImage, 0, 0, absentRectangleArea, GraphicsUnit.Pixel );
                }

                currentTile.Save(@"e:\Tests\Tile.bmp");
            }
        }
    }

5

มันค่อนข้างง่าย:

  • สร้างBitmapวัตถุใหม่ด้วยขนาดที่ถูกครอบตัด
  • ใช้Graphics.FromImageเพื่อสร้างGraphicsวัตถุสำหรับบิตแมปใหม่
  • ใช้DrawImageวิธีการวาดภาพลงบนบิตแมปด้วยพิกัด X และ Y เชิงลบ

5

นี่คือตัวอย่างง่ายๆของการครอบตัดรูปภาพ

public Image Crop(string img, int width, int height, int x, int y)
{
    try
    {
        Image image = Image.FromFile(img);
        Bitmap bmp = new Bitmap(width, height, PixelFormat.Format24bppRgb);
        bmp.SetResolution(80, 60);

        Graphics gfx = Graphics.FromImage(bmp);
        gfx.SmoothingMode = SmoothingMode.AntiAlias;
        gfx.InterpolationMode = InterpolationMode.HighQualityBicubic;
        gfx.PixelOffsetMode = PixelOffsetMode.HighQuality;
        gfx.DrawImage(image, new Rectangle(0, 0, width, height), x, y, width, height, GraphicsUnit.Pixel);
        // Dispose to free up resources
        image.Dispose();
        bmp.Dispose();
        gfx.Dispose();

        return bmp;
    }
    catch (Exception ex)
    {
        MessageBox.Show(ex.Message);
        return null;
    }            
}

5
เขาเป็นคนเดียวที่กล่าวถึงความละเอียดวิธีการทั้งหมดข้างต้นจะล้มเหลวหากภาพต้นฉบับมีความละเอียดที่ไม่ได้มาตรฐาน
net_prog

1
ใช้ bmp.SetResolution (รูปภาพ. Horizo ​​nRRolution, รูปภาพ. แนวตั้ง Resolution); เพื่อแก้ไขปัญหาการแก้ไข
Morbia

2
ยกเว้นกรณีนี้จะทำให้อิมเมจรั่วไหล, วัตถุ bmp และ gfx ทำไมไม่ห่อในการใช้คำสั่ง?
Darius Kucinskas


2

ฉันกำลังมองหาฟังก์ชั่นที่ง่ายและรวดเร็วโดยไม่มี libary เพิ่มเติมเพื่อทำงาน ฉันลองวิธีแก้ปัญหาของ Nicksแต่ฉันต้องการ 29,4 วินาทีในการ "แยก" รูปภาพของไฟล์ atlas 1195 ภาพ ดังนั้นในภายหลังฉันจัดการด้วยวิธีนี้และต้องการ 2,43 วินาทีเพื่อทำงานเดียวกัน บางทีนี่อาจจะเป็นประโยชน์

// content of the Texture class
public class Texture
{
    //name of the texture
    public string name { get; set; }
    //x position of the texture in the atlas image
    public int x { get; set; }
    //y position of the texture in the atlas image
    public int y { get; set; }
    //width of the texture in the atlas image
    public int width { get; set; }
    //height of the texture in the atlas image
    public int height { get; set; }
}

Bitmap atlasImage = new Bitmap(@"C:\somepicture.png");
PixelFormat pixelFormat = atlasImage.PixelFormat;

foreach (Texture t in textureList)
{
     try
     {
           CroppedImage = new Bitmap(t.width, t.height, pixelFormat);
           // copy pixels over to avoid antialiasing or any other side effects of drawing
           // the subimages to the output image using Graphics
           for (int x = 0; x < t.width; x++)
               for (int y = 0; y < t.height; y++)
                   CroppedImage.SetPixel(x, y, atlasImage.GetPixel(t.x + x, t.y + y));
           CroppedImage.Save(Path.Combine(workingFolder, t.name + ".png"), ImageFormat.Png);
     }
     catch (Exception ex)
     {
          // handle the exception
     }
}

1

การครอบตัดรูปภาพเป็นเรื่องง่ายมากใน C # อย่างไรก็ตามการทำสิ่งที่คุณจะจัดการการครอบตัดภาพของคุณจะยากขึ้นเล็กน้อย

ตัวอย่างด้านล่างเป็นวิธีการครอบตัดรูปภาพใน C #

var filename = @"c:\personal\images\horizon.png";
var img = Image.FromFile(filename);
var rect = new Rectangle(new Point(0, 0), img.Size);
var cloned = new Bitmap(img).Clone(rect, img.PixelFormat);
var bitmap = new Bitmap(cloned, new Size(50, 50));
cloned.Dispose();

1

มีตัวห่อหุ้ม C # สำหรับสิ่งที่เป็นโอเพนซอร์สซึ่งโฮสต์อยู่บน Codeplex ชื่อWeb Image Cropping

ลงทะเบียนตัวควบคุม

<%@ Register Assembly="CS.Web.UI.CropImage" Namespace="CS.Web.UI" TagPrefix="cs" %>

ปรับขนาด

<asp:Image ID="Image1" runat="server" ImageUrl="images/328.jpg" />
<cs:CropImage ID="wci1" runat="server" Image="Image1" 
     X="10" Y="10" X2="50" Y2="50" />

การครอบตัดโค้ดด้านหลัง - วิธีการครอบตัดการโทรเมื่อคลิกปุ่มตัวอย่าง;

wci1.Crop(Server.MapPath("images/sample1.jpg"));


0

สมมติว่าคุณหมายความว่าคุณต้องการใช้ไฟล์ภาพ (JPEG, BMP, TIFF, ฯลฯ ) และครอบตัดแล้วบันทึกเป็นไฟล์ภาพขนาดเล็กลงฉันขอแนะนำให้ใช้เครื่องมือของบุคคลที่สามที่มี. NET API นี่คือตัวอย่างบางส่วนที่ได้รับความนิยมที่ฉันชอบ:

LeadTools
Accusoft Pegasus Snowbound Imaging SDK


0

ตัวอย่างนี้เท่านั้นที่ทำงานได้โดยไม่มีปัญหา:

var crop = new Rectangle(0, y, bitmap.Width, h);
var bmp = new Bitmap(bitmap.Width, h);
var tempfile = Application.StartupPath+"\\"+"TEMP"+"\\"+Path.GetRandomFileName();


using (var gr = Graphics.FromImage(bmp))
{
    try
    {
        var dest = new Rectangle(0, 0, bitmap.Width, h);
        gr.DrawImage(image,dest , crop, GraphicsUnit.Point);
        bmp.Save(tempfile,ImageFormat.Jpeg);
        bmp.Dispose();
    }
    catch (Exception)
    {


    }

}

0

นี่เป็นอีกวิธีหนึ่ง ในกรณีของฉันฉันมี:

  • 2 ตัวควบคุม updown ตัวเลข (เรียกว่า LeftMargin และ TopMargin)
  • 1 กล่องรูปภาพ (pictureBox1)
  • 1 ปุ่มที่ฉันเรียกว่าสร้าง
  • 1 ภาพใน C: \ imagenes \ myImage.gif

ข้างในปุ่มฉันมีรหัสนี้:

Image myImage = Image.FromFile(@"C:\imagenes\myImage.gif");
Bitmap croppedBitmap = new Bitmap(myImage);
croppedBitmap = croppedBitmap.Clone(
            new Rectangle(
                (int)LeftMargin.Value, (int)TopMargin.Value,
                myImage.Width - (int)LeftMargin.Value,
                myImage.Height - (int)TopMargin.Value),
            System.Drawing.Imaging.PixelFormat.DontCare);
pictureBox1.Image = croppedBitmap;

ฉันลองใช้ Visual Studio 2012 ด้วย C # ฉันพบโซลูชันนี้จากหน้านี้


0

นี่คือการสาธิตการทำงานบน GitHub

https://github.com/SystematixIndore/Crop-SaveImageInCSharp

<%@ Page Language="C#" AutoEventWireup="true" CodeBehind="WebForm1.aspx.cs" Inherits="WebApplication1.WebForm1" %>

<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml">
<head runat="server">
  <title></title>
 <link href="css/jquery.Jcrop.css" rel="stylesheet" type="text/css" />
<script type="text/javascript" src="http://ajax.googleapis.com/ajax/libs/jquery/1.3/jquery.min.js"></script>
<script type="text/javascript" src="js/jquery.Jcrop.js"></script>
</head>
<body>
  <form id="form2" runat="server">
  <div>
    <asp:Panel ID="pnlUpload" runat="server">
      <asp:FileUpload ID="Upload" runat="server" />
      <br />
      <asp:Button ID="btnUpload" runat="server" OnClick="btnUpload_Click" Text="Upload" />
      <asp:Label ID="lblError" runat="server" Visible="false" />
    </asp:Panel>
    <asp:Panel ID="pnlCrop" runat="server" Visible="false">
      <asp:Image ID="imgCrop" runat="server" />
      <br />
      <asp:HiddenField ID="X" runat="server" />
      <asp:HiddenField ID="Y" runat="server" />
      <asp:HiddenField ID="W" runat="server" />
      <asp:HiddenField ID="H" runat="server" />
      <asp:Button ID="btnCrop" runat="server" Text="Crop" OnClick="btnCrop_Click" />
    </asp:Panel>
    <asp:Panel ID="pnlCropped" runat="server" Visible="false">
      <asp:Image ID="imgCropped" runat="server" />
    </asp:Panel>
  </div>
  </form>
    <script type="text/javascript">
  jQuery(document).ready(function() {
    jQuery('#imgCrop').Jcrop({
      onSelect: storeCoords
    });
  });

  function storeCoords(c) {
    jQuery('#X').val(c.x);
    jQuery('#Y').val(c.y);
    jQuery('#W').val(c.w);
    jQuery('#H').val(c.h);
  };

</script>
</body>
</html>

ตรรกะรหัส C # สำหรับการอัปโหลดและครอบตัด

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.UI;
using System.Web.UI.WebControls;
using System.IO;
using SD = System.Drawing;
using System.Drawing.Imaging;
using System.Drawing.Drawing2D;

namespace WebApplication1
{
    public partial class WebForm1 : System.Web.UI.Page
    {
        String path = HttpContext.Current.Request.PhysicalApplicationPath + "images\\";
        protected void Page_Load(object sender, EventArgs e)
        {

        }
        protected void btnUpload_Click(object sender, EventArgs e)
        {
            Boolean FileOK = false;
            Boolean FileSaved = false;

            if (Upload.HasFile)
            {
                Session["WorkingImage"] = Upload.FileName;
                String FileExtension = Path.GetExtension(Session["WorkingImage"].ToString()).ToLower();
                String[] allowedExtensions = { ".png", ".jpeg", ".jpg", ".gif" };
                for (int i = 0; i < allowedExtensions.Length; i++)
                {
                    if (FileExtension == allowedExtensions[i])
                    {
                        FileOK = true;
                    }
                }
            }

            if (FileOK)
            {
                try
                {
                    Upload.PostedFile.SaveAs(path + Session["WorkingImage"]);
                    FileSaved = true;
                }
                catch (Exception ex)
                {
                    lblError.Text = "File could not be uploaded." + ex.Message.ToString();
                    lblError.Visible = true;
                    FileSaved = false;
                }
            }
            else
            {
                lblError.Text = "Cannot accept files of this type.";
                lblError.Visible = true;
            }

            if (FileSaved)
            {
                pnlUpload.Visible = false;
                pnlCrop.Visible = true;
                imgCrop.ImageUrl = "images/" + Session["WorkingImage"].ToString();
            }
        }

        protected void btnCrop_Click(object sender, EventArgs e)
        {
            string ImageName = Session["WorkingImage"].ToString();
            int w = Convert.ToInt32(W.Value);
            int h = Convert.ToInt32(H.Value);
            int x = Convert.ToInt32(X.Value);
            int y = Convert.ToInt32(Y.Value);

            byte[] CropImage = Crop(path + ImageName, w, h, x, y);
            using (MemoryStream ms = new MemoryStream(CropImage, 0, CropImage.Length))
            {
                ms.Write(CropImage, 0, CropImage.Length);
                using (SD.Image CroppedImage = SD.Image.FromStream(ms, true))
                {
                    string SaveTo = path + "crop" + ImageName;
                    CroppedImage.Save(SaveTo, CroppedImage.RawFormat);
                    pnlCrop.Visible = false;
                    pnlCropped.Visible = true;
                    imgCropped.ImageUrl = "images/crop" + ImageName;
                }
            }
        }

        static byte[] Crop(string Img, int Width, int Height, int X, int Y)
        {
            try
            {
                using (SD.Image OriginalImage = SD.Image.FromFile(Img))
                {
                    using (SD.Bitmap bmp = new SD.Bitmap(Width, Height))
                    {
                        bmp.SetResolution(OriginalImage.HorizontalResolution, OriginalImage.VerticalResolution);
                        using (SD.Graphics Graphic = SD.Graphics.FromImage(bmp))
                        {
                            Graphic.SmoothingMode = SmoothingMode.AntiAlias;
                            Graphic.InterpolationMode = InterpolationMode.HighQualityBicubic;
                            Graphic.PixelOffsetMode = PixelOffsetMode.HighQuality;
                            Graphic.DrawImage(OriginalImage, new SD.Rectangle(0, 0, Width, Height), X, Y, Width, Height, SD.GraphicsUnit.Pixel);
                            MemoryStream ms = new MemoryStream();
                            bmp.Save(ms, OriginalImage.RawFormat);
                            return ms.GetBuffer();
                        }
                    }
                }
            }
            catch (Exception Ex)
            {
                throw (Ex);
            }
        }
    }
}
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.