欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

深入c# GDI+簡單繪圖的具體操作步驟(三)

 更新時間:2013年05月20日 15:40:02   作者:  
前兩篇已經基本向大家介紹了繪圖的基本知識.那么,我就用我們上兩篇所學的,做幾個例子.我們先來做一個簡單的--仿QQ截圖
關于這個的例子其實網上已經有這方面的資料了,但是為了文章的完整性,還是覺得有必要講解.
我們先來看一下效果:
  
                                                                                                                  (圖(圖1)
                                                      20083191642.JPG

( 圖2 )

接下來看看這是如何做到的. 
思路:聊天窗體上有一個截圖按鈕,點擊按鈕后,程序將整個屏幕畫在一個新的全屏窗體上,然后顯示這個窗體.因為是全屏的窗體,并且隱藏了菜單欄、工具欄等,所以在我們看來就好像是一個桌面的截圖,然后在這個新窗體上畫矩形,最后保存矩形中的內容并顯示在原來的聊天窗體中.
步驟:
A.新建一個窗體.命名為Catch.然后設置這個窗體的FormBorderStyle為None,WindowState為Maximized.
B.我們對代碼進行編輯:
復制代碼 代碼如下:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Text;
using System.Windows.Forms;
namespace Client
{
    public partial class Catch : Form
    {
        public Catch()
        {
            InitializeComponent();
        }
        用戶變量#region 用戶變量
        private Point DownPoint = Point.Empty;//記錄鼠標按下坐標,用來確定繪圖起點
        private bool CatchFinished = false;//用來表示是否截圖完成
        private bool CatchStart = false;//表示截圖開始
        private Bitmap originBmp;//用來保存原始圖像
        private Rectangle CatchRect;//用來保存截圖的矩形
        #endregion
        //窗體初始化操作
        private void Catch_Load(object sender, EventArgs e)
        {
            this.SetStyle(ControlStyles.OptimizedDoubleBuffer | ControlStyles.AllPaintingInWmPaint | ControlStyles.UserPaint, true);
            this.UpdateStyles();
            //以上兩句是為了設置控件樣式為雙緩沖,這可以有效減少圖片閃爍的問題,關于這個大家可以自己去搜索下
            originBmp = new Bitmap(this.BackgroundImage);//BackgroundImage為全屏圖片,我們另用變量來保存全屏圖片
        }
        //鼠標右鍵點擊結束截圖
        private void Catch_MouseClick(object sender, MouseEventArgs e)
        {
            if (e.Button == MouseButtons.Right)
            {
                this.DialogResult = DialogResult.OK;
                this.Close();
            }
        }
        //鼠標左鍵按下時動作
        private void Catch_MouseDown(object sender, MouseEventArgs e)
        {
            if (e.Button == MouseButtons.Left)
            {
                if (!CatchStart)
                {//如果捕捉沒有開始
                    CatchStart = true;
                    DownPoint = new Point(e.X, e.Y);//保存鼠標按下坐標
                }
            }
        }
        private void Catch_MouseMove(object sender, MouseEventArgs e)
        {
            if (CatchStart)
            {//如果捕捉開始
                Bitmap destBmp = (Bitmap)originBmp.Clone();//新建一個圖片對象,并讓它與原始圖片相同
                Point newPoint = new Point(DownPoint.X, DownPoint.Y);//獲取鼠標的坐標
                Graphics g = Graphics.FromImage(destBmp);//在剛才新建的圖片上新建一個畫板
                Pen p = new Pen(Color.Blue,1);
                int width = Math.Abs(e.X - DownPoint.X), height = Math.Abs(e.Y - DownPoint.Y);//獲取矩形的長和寬
                if (e.X < DownPoint.X)
                {
                    newPoint.X = e.X;
                }
                if (e.Y < DownPoint.Y)
                {
                    newPoint.Y = e.Y;
                }
                CatchRect = new Rectangle(newPoint,new Size(width,height));//保存矩形
                g.DrawRectangle(p,CatchRect);//將矩形畫在這個畫板上
                g.Dispose();//釋放目前的這個畫板
                p.Dispose();
                Graphics g1 = this.CreateGraphics();//重新新建一個Graphics類
                //如果之前那個畫板不釋放,而直接g=this.CreateGraphics()這樣的話無法釋放掉第一次創(chuàng)建的g,因為只是把地址轉到新的g了.如同string一樣
                g1 = this.CreateGraphics();//在整個全屏窗體上新建畫板
                g1.DrawImage(destBmp,new Point(0,0));//將剛才所畫的圖片畫到這個窗體上
                //這個也可以屬于二次緩沖技術,如果直接將矩形畫在窗體上,會造成圖片抖動并且會有無數個矩形.
                g1.Dispose();
                destBmp.Dispose();//要及時釋放,不然內存將會被大量消耗

            }
        }
        private void Catch_MouseUp(object sender, MouseEventArgs e)
        {
            if (e.Button == MouseButtons.Left)
            {
                if (CatchStart)
                {
                    CatchStart = false;
                    CatchFinished = true;

                }
            }
        }
        //鼠標雙擊事件,如果鼠標位于矩形內,則將矩形內的圖片保存到剪貼板中
        private void Catch_MouseDoubleClick(object sender, MouseEventArgs e)
        {
            if (e.Button == MouseButtons.Left&&CatchFinished)
            {
                if (CatchRect.Contains(new Point(e.X, e.Y)))
                {
                    Bitmap CatchedBmp = new Bitmap(CatchRect.Width, CatchRect.Height);//新建一個于矩形等大的空白圖片
                    Graphics g = Graphics.FromImage(CatchedBmp);
                    g.DrawImage(originBmp, new Rectangle(0, 0, CatchRect.Width, CatchRect.Height), CatchRect, GraphicsUnit.Pixel);
                    //把orginBmp中的指定部分按照指定大小畫在畫板上
                    Clipboard.SetImage(CatchedBmp);//將圖片保存到剪貼板
                    g.Dispose();
                    CatchFinished = false;
                    this.BackgroundImage = originBmp;
                    CatchedBmp.Dispose();
                    this.DialogResult = DialogResult.OK;
                    this.Close();
                }
            }
        }
    }
}

C.創(chuàng)建了Catch窗體后,我們在截圖按鈕(位于聊天窗體上)上加入以下事件:
復制代碼 代碼如下:

 private void bCatch_Click(object sender, EventArgs e)
        {
            if (bCatch_HideCurrent.Checked)
            {
                this.Hide();//隱藏當前窗體
                Thread.Sleep(50);//讓線程睡眠一段時間,窗體消失需要一點時間
                Catch CatchForm = new Catch();
                Bitmap CatchBmp = new Bitmap(Screen.AllScreens[0].Bounds.Width, Screen.AllScreens[0].Bounds.Height);//新建一個和屏幕大小相同的圖片        
                Graphics g = Graphics.FromImage(CatchBmp);
                g.CopyFromScreen(new Point(0, 0), new Point(0, 0), new Size(Screen.AllScreens[0].Bounds.Width, Screen.AllScreens[0].Bounds.Height));//保存全屏圖片
                CatchForm.BackgroundImage = CatchBmp;//將Catch窗體的背景設為全屏時的圖片
                if (CatchForm.ShowDialog() == DialogResult.OK)
                {//如果Catch窗體結束,就將剪貼板中的圖片放到信息發(fā)送框中
                    IDataObject iData = Clipboard.GetDataObject();
                    DataFormats.Format myFormat = DataFormats.GetFormat(DataFormats.Bitmap);
                    if (iData.GetDataPresent(DataFormats.Bitmap))
                    {
                        richtextbox1.Paste(myFormat);
                        Clipboard.Clear();//清除剪貼板中的對象
                    }
                    this.Show();//重新顯示窗體
                }
            }
        }

這樣我們的截圖功能便完成了.
我想對于初學者來說如何消去第一次繪制的圖片是個比較困難的問題.如果沒有采取措施,你會發(fā)現只要你鼠標移動,就會畫一個矩形,這樣便會出現N多的矩形,而我們只是要最后的那一個.
一般解決這種問題的方法有兩種:
1.就是在繪制第二個圖形時,我們先用與底色相同的顏色將上次繪制的圖形重新繪制一下.但這往往需要底色為純色時使用.
2.我們并不直接將圖形畫在畫板上,我們用一個圖片A來保存原畫板上的圖片.然后再新建一個與圖片A相同的圖片B,將我們要繪制的圖形畫在該圖片B上,然后再將該圖片B畫在畫板上.這樣圖片A并沒有被改變.于是第二次畫的時候我們還是同樣新建一個與圖片A相同的圖片進行繪制.那么上一次的圖形就不會被保留下來.問題也就解決了.

相關文章

  • Unity?UGUI的Canvas畫布組件使用示例詳解

    Unity?UGUI的Canvas畫布組件使用示例詳解

    這篇文章主要介紹了Unity?UGUI的Canvas畫布組件使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-07-07
  • C#基于UDP進行異步通信的方法

    C#基于UDP進行異步通信的方法

    這篇文章主要介紹了C#基于UDP進行異步通信的方法,實例分析了C#基于UDP實現異步通信的相關技巧,需要的朋友可以參考下
    2015-04-04
  • 深入理解C#中回調函數

    深入理解C#中回調函數

    回調函數是一種在編程中常用的概念,本文將介紹回調函數的概念、語法和應用,并討論如何設計優(yōu)化和重用回調函數,以及它們在并發(fā)編程中的用途,感興趣的可以了解一下
    2024-02-02
  • C#啟動進程的幾種常用方法

    C#啟動進程的幾種常用方法

    這篇文章主要介紹了C#啟動進程的幾種常用方法,實例分析了C#對系統(tǒng)進行的相關操作技巧,需要的朋友可以參考下
    2015-05-05
  • 基于C#的圖表控件庫 ScottPlot編譯visual studio 2022

    基于C#的圖表控件庫 ScottPlot編譯visual studio 2022

    基于 C# 的 圖表控件庫 ScottPlot,開源免費,可以用于開發(fā)一些上位機軟件,如電壓、電流波形的顯示,開發(fā)【示波器】圖形界面,可以顯示一些圖表、波形,總之功能比較的強大,本文介紹了基于C#的圖表控件庫 ScottPlot編譯visual studio 2022,需要的朋友可以參考下
    2022-06-06
  • 使用C#實現阿拉伯數字到大寫中文的轉換

    使用C#實現阿拉伯數字到大寫中文的轉換

    這篇文章主要介紹了C#實現阿拉伯數字轉為大寫中文的實現代碼,需要的朋友可以參考下
    2007-03-03
  • C#修改MAC地址類的實例

    C#修改MAC地址類的實例

    C#修改MAC地址類的實例,需要的朋友可以參考一下
    2013-03-03
  • unity實現鼠標拖住3D物體

    unity實現鼠標拖住3D物體

    這篇文章主要為大家詳細介紹了unity實現鼠標拖住3D物體,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-07-07
  • 簡單談談C#中深拷貝、淺拷貝

    簡單談談C#中深拷貝、淺拷貝

    深拷貝和淺拷貝都是對對象進行的拷貝復制。而他們的不同就在于:一個對象中可能會有值類型的字段,也可能會有引用類型的字段。淺拷貝直接拷貝值、引用類型的指針。而深拷貝則是重新構造字段指針所指向的對象,并沒有拷貝該字段的指針,或是與該字段指針指向同一塊內存
    2015-06-06
  • C#字體池技術實現代碼詳解

    C#字體池技術實現代碼詳解

    在本篇文章里小編給大家整理的是關于C#字體池技術實現代碼詳解內容,有需要的朋友們可以學習下。
    2019-11-11

最新評論