最近做一個圖象的采集,需要一個圖形的選擇控件,但是在.net下沒有類似vb中的shape控件,所以考慮了自己寫一個控件。
下面我將從頭創(chuàng)建控件,這個控件主要是用來選擇圖形的rectangle,有一下幾個屬性color bordercolor:邊框顏色,color backcolor:背景顏色,bool resizeble:是否可移動, rectangle selectrectangle:選擇區(qū)域。
打開vs2003(我用的這個版本),新建一個c#控件庫,ok,拷貝如下代碼到你的代碼里。
using system;
using system.collections;
using system.componentmodel;
using system.drawing;
using system.data;
using system.windows.forms;
namespace windowsextendedcontrols
{
 /// <summary>
 /// 控件
 /// </summary>
 public class shapeex : system.windows.forms.control
 {
  /// <summary>
  /// 必需的設(shè)計器變量。
  /// </summary>
  /// 
  private color _bordercolor=new color();
  private color _backcolor=new color();
  private bool _resizeble;
  private point _selflocation=new point();
  private point _mouselocation=new point();
  private int _selfwidth;
  private int _selfheight;
  private int _selectselctedindex;//0-8,0:sizeall
  private rectangle _rectleftselector=new rectangle();
  private rectangle _recttopselector=new rectangle();
  private rectangle _rectrightselector=new rectangle();
  private rectangle _rectbottomselector=new rectangle();
  private rectangle _rectlefttopselector=new rectangle();
  private rectangle _rectrighttopselector=new rectangle();
  private rectangle _rectrightbottomselector=new rectangle();
  private rectangle _rectleftbottomselector=new rectangle();
  private system.componentmodel.container components = null;
  public shapeex()
  {
   // 該調(diào)用是 windows.forms 窗體設(shè)計器所必需的。
   initializecomponent();
// todo: 在 initcomponent 調(diào)用后添加任何初始化
  }
  [defaultvalue("black"),description("邊框顏色"),category("appearance")] 
  public color bordercolor
  {
   get 
   {
    // insert code here.
    return _bordercolor;
   }
   set 
   {
    _bordercolor=value;
    this.invalidate();
   }
  }
  [defaultvalue("control"),description("背景顏色"),category("appearance")] 
  public override color backcolor
  {
   get 
   {
    // insert code here.
    return _backcolor;
   }
   set 
   {
    _backcolor=value;
    this.invalidate();
   }
  }
  [defaultvalue(false),description("運(yùn)行中控件大小是否可拖拽編輯"),category("behavior")] 
  public  bool resizeble
  {
   get 
   {
    // insert code here.
    return _resizeble;
   }
   set 
   {
    _resizeble=value;
    this.invalidate();
   }
  }
  [description("控件選擇區(qū)域"),category("behavior")] 
  public  rectangle selectrectangle
  {
   get 
   {
    rectangle selectrectangler=new rectangle();
    selectrectangler.x = this.location.x+7;
    selectrectangler.y = this.location.y+7;
    selectrectangler.height = this.height-15;
    selectrectangler.width = this.width-15;
    return selectrectangler;
   }
  
  }
 
  protected override void onpaint(painteventargs pe)
  {
   // calling the base class onpaint
   base.onpaint(pe);
   redrawcontrol(pe.graphics);
  }
  private void drawselector(graphics graphics)
  {
   solidbrush selectorpen=new solidbrush(color.white);
   pen borderpen=new pen(this._bordercolor,1);
   try
   {
    //實心
    pointf[] leftpoints=getpointf(0,this.height/2-3,6,6);
    graphics.fillclosedcurve(selectorpen, leftpoints);
    pointf[] toppoints=getpointf(this.width/2-3,0,6,6);
    graphics.fillclosedcurve(selectorpen, toppoints);
    pointf[] rightpoints=getpointf(this.width-7,this.height/2-3,6,6);
    graphics.fillclosedcurve(selectorpen, rightpoints);
    pointf[] bottompoints=getpointf(this.width/2-3,this.height-7,6,6);
    graphics.fillclosedcurve(selectorpen, bottompoints);
    pointf[] lefttoppoints=getpointf(0,0,6,6);
    graphics.fillclosedcurve(selectorpen, lefttoppoints);
    pointf[] righttoppoints=getpointf(this.width-7,0,6,6);
    graphics.fillclosedcurve(selectorpen, righttoppoints);
    pointf[] rightbottompoints=getpointf(this.width-7,this.height-7,6,6);
    graphics.fillclosedcurve(selectorpen, rightbottompoints);
    
    pointf[] leftbottompoints=getpointf(0,this.height-7,6,6);
    graphics.fillclosedcurve(selectorpen, leftbottompoints);
    //邊框
    _rectleftselector.x = 0;
    _rectleftselector.y = this.height/2-3;
    _rectleftselector.height = 6;
    _rectleftselector.width = 6;
    graphics.drawrectangle(borderpen, _rectleftselector);
    _recttopselector.x = this.width/2-3;
    _recttopselector.y = 0;
    _recttopselector.height = 6;
    _recttopselector.width = 6;
    graphics.drawrectangle(borderpen, _recttopselector);
    _rectrightselector.x = this.width-7;
    _rectrightselector.y = this.height/2-3;
    _rectrightselector.height = 6;
    _rectrightselector.width = 6;
    graphics.drawrectangle(borderpen, _rectrightselector);
    _rectbottomselector.x = this.width/2-3;
    _rectbottomselector.y = this.height-7;
    _rectbottomselector.height = 6;
    _rectbottomselector.width = 6;
    graphics.drawrectangle(borderpen, _rectbottomselector);
    _rectlefttopselector.x=0;
    _rectlefttopselector.y=0;
    _rectlefttopselector.width=6;
    _rectlefttopselector.height=6;
    graphics.drawrectangle(borderpen, _rectlefttopselector);
    _rectrighttopselector.x=this.width-7;
    _rectrighttopselector.y=0;
    _rectrighttopselector.width=6;
    _rectrighttopselector.height=6;
    graphics.drawrectangle(borderpen, _rectrighttopselector);
    _rectrightbottomselector.x=this.width-7;
    _rectrightbottomselector.y=this.height-7;
    _rectrightbottomselector.width=6;
    _rectrightbottomselector.height=6;
    graphics.drawrectangle(borderpen, _rectrightbottomselector);
    _rectleftbottomselector.x=0;
    _rectleftbottomselector.y=this.height-7;
    _rectleftbottomselector.width=6;
    _rectleftbottomselector.height=6;
    graphics.drawrectangle(borderpen, _rectleftbottomselector);
   }
   catch(exception e)
   {
    throw e;
   }
   finally
   {
    selectorpen.dispose();
    borderpen.dispose();
   }
    
  }
  private void redrawcontrol(graphics graphics)
  {
   
   try
   {
   
    //繪制背景
    /*
    graphics.clear(this._backcolor);
    solidbrush backpen=new solidbrush(this._backcolor);
    pointf point1 = new pointf(1,1);
    pointf point2 = new pointf(this.width-2,1);
    pointf point3 = new pointf(this.width-2,this.height-2);
    pointf point4 = new pointf(1,this.height-2);
    pointf[] points = {point1, point2, point3, point4};
    graphics.fillclosedcurve(backpen, points);
    */
    //繪制邊框    
    rectangle rectborder=new rectangle();
    pen borderpen=new pen(this._bordercolor,1);
    rectborder.x = 7;
    rectborder.y = 7;
    rectborder.height = this.height-15;
    rectborder.width = this.width-15;
    graphics.drawrectangle(borderpen, rectborder);
    //繪制編輯框
    if (_resizeble)
    {
     drawselector(graphics);
    }
   }
   catch(exception e)
   {
    throw e;
   }
   finally
   {
    graphics.dispose();
   }
  }
  /// <summary>
  /// 清理所有正在使用的資源。
  /// </summary>
  private pointf[] getpointf(int x,int y,int width,int height){
   pointf point1 = new pointf(x,y);
   pointf point2 = new pointf(x+width,y);
   pointf point3 = new pointf(x+width,y+height);
   pointf point4 = new pointf(x,y+height);
   pointf[] points = {point1, point2, point3, point4};
   return points;
  }
  protected override void dispose( bool disposing )
  {
   if( disposing )
   {
    if( components != null )
     components.dispose();
   }
   base.dispose( disposing );
  }
  #region 組件設(shè)計器生成的代碼
  /// <summary>
  /// 設(shè)計器支持所需的方法 - 不要使用代碼編輯器 
  /// 修改此方法的內(nèi)容。
  /// </summary>
  private void initializecomponent()
  {
   components = new system.componentmodel.container();
   this.resize+=new eventhandler(shapeex_resize);
   this.mousedown+=new mouseeventhandler(shapeex_mousedown);
   this.mousemove+=new mouseeventhandler(shapeex_mousemove);
   this.mouseleave+=new eventhandler(shapeex_mouseleave);
   this.mouseup+=new mouseeventhandler(shapeex_mouseup);
   this._bordercolor=color.black;
   this._backcolor=color.fromname("control");
   this._resizeble=false;
   this._selectselctedindex=-1;
   setstyle(controlstyles.supportstransparentbackcolor, true);
  }
  #endregion
  private void shapeex_resize(object sender, eventargs e)
  {
   if (this.width<16 || this.height<16)
   {
    this.width=16;
    this.height=16;
   }
   this.invalidate();
  }
  private void shapeex_mousedown(object sender, mouseeventargs e)
  {
   if (_resizeble)
   {
    if (_rectleftselector.contains(e.x,e.y) || _rectrightselector.contains(e.x,e.y) || _recttopselector.contains(e.x,e.y) || _rectbottomselector.contains(e.x,e.y) ||_rectlefttopselector.contains(e.x,e.y) || _rectrighttopselector.contains(e.x,e.y) || _rectrightbottomselector.contains(e.x,e.y) || _rectleftbottomselector.contains(e.x,e.y))
    {
     if (_rectlefttopselector.contains(e.x,e.y) )
     {
      this.cursor=cursors.sizenwse;
      this._selectselctedindex=1;
     }
     
     if (_recttopselector.contains(e.x,e.y) )
     {
      this.cursor=cursors.sizens;
      this._selectselctedindex=2;
     }
     if (_rectrighttopselector.contains(e.x,e.y) )
     {
      this.cursor=cursors.sizenesw;
      this._selectselctedindex=3;
     }
     if (_rectrightselector.contains(e.x,e.y) )
     {
      this.cursor=cursors.sizewe;
      this._selectselctedindex=4;
     }
     if (_rectrightbottomselector.contains(e.x,e.y) )
     {
      this.cursor=cursors.sizenwse;
      this._selectselctedindex=5;
     }
     if (_rectbottomselector.contains(e.x,e.y))
     {
      this.cursor=cursors.sizens;
      this._selectselctedindex=6;
     }
     if (_rectleftbottomselector.contains(e.x,e.y) )
     {
      this.cursor=cursors.sizenesw;
      this._selectselctedindex=7;
     }
     if (_rectleftselector.contains(e.x,e.y))
     {
      this.cursor=cursors.sizewe;
      this._selectselctedindex=8;
     }
    }
    else
    {
     this.cursor=cursors.sizeall;
     this._selectselctedindex=0;
    }
    this._selflocation.x=this.location.x;
    this._selflocation.y=this.location.y;
    this._mouselocation.x=cursor.position.x;
    this._mouselocation.y=cursor.position.y;
    this._selfwidth=this.width;
    this._selfheight=this.height;
   }
  }
  private void shapeex_mousemove(object sender, mouseeventargs e)
  {
   //move and resize
   switch (this._selectselctedindex)
   {
    case 0:
     this.location=new point(cursor.position.x-(_mouselocation.x-_selflocation.x),cursor.position.y-(_mouselocation.y-_selflocation.y));
     break;
    case 1:
     this.height=this._selfheight-(cursor.position.y-_mouselocation.y);
     this.width=this._selfwidth-(cursor.position.x-_mouselocation.x);
     this.location=new point(cursor.position.x-_mouselocation.x+_selflocation.x,cursor.position.y-_mouselocation.y+_selflocation.y);
     break;
    case 2:
     this.height=this._selfheight-(cursor.position.y-_mouselocation.y);
     this.location=new point(_selflocation.x,cursor.position.y-_mouselocation.y+_selflocation.y);
     break;
    case 3:
     this.height=this._selfheight-(cursor.position.y-_mouselocation.y);
     this.width=this._selfwidth+(cursor.position.x-_mouselocation.x);
     this.location=new point(_selflocation.x,cursor.position.y-(_mouselocation.y-_selflocation.y));
     break;
    case 4:
     this.width=this._selfwidth+(cursor.position.x-_mouselocation.x);
     break;
    case 5:
     this.height=this._selfheight+(cursor.position.y-_mouselocation.y);
     this.width=this._selfwidth+(cursor.position.x-_mouselocation.x);
     break;
    case 6:
     this.height=this._selfheight+(cursor.position.y-_mouselocation.y);
     break;
    case 7:
     this.height=this._selfheight+(cursor.position.y-_mouselocation.y);
     this.width=this._selfwidth-(cursor.position.x-_mouselocation.x);
     this.location=new point(cursor.position.x-_mouselocation.x+_selflocation.x,_selflocation.y);
     break;
    case 8:
     this.width=this._selfwidth-(cursor.position.x-_mouselocation.x);
     this.location=new point(cursor.position.x-_mouselocation.x+_selflocation.x,_selflocation.y);
     break;
   }
}
  private void shapeex_mouseleave(object sender, eventargs e)
  {
   this.cursor=cursors.default;
   this._selectselctedindex=-1;
  }
  private void shapeex_mouseup(object sender, mouseeventargs e)
  {
   this.cursor=cursors.default;
   this._selectselctedindex=-1;
  }
 }
}
 
下面講一下控件具體如何工作,首先要寫他的屬性以及重寫他的屬性,
private color _bordercolor=new color();
 [defaultvalue("black"),description("邊框顏色"),category("appearance")] 
  public color bordercolor
  {
   get 
   {
    // insert code here.
    return _bordercolor;
   }
   set 
   {
    _bordercolor=value;
    this.invalidate();
   }
  }
defaultvalue:設(shè)定默認(rèn)值,description:描述,就是屬性下面的說明,category:屬性分類.其他的同理
設(shè)置好屬性以后應(yīng)該重寫他的繪制過程,也就是
protected override void onpaint(painteventargs pe)
  {
   // calling the base class onpaint
   base.onpaint(pe);
   redrawcontrol(pe.graphics);
  }
 private void redrawcontrol(graphics graphics)
  {
   
   try
   {
    //繪制邊框    
    rectangle rectborder=new rectangle();
    pen borderpen=new pen(this._bordercolor,1);
    rectborder.x = 7;
    rectborder.y = 7;
    rectborder.height = this.height-15;
    rectborder.width = this.width-15;
    graphics.drawrectangle(borderpen, rectborder);
    //繪制編輯框
    if (_resizeble)
    {
     drawselector(graphics);
    }
   }
   catch(exception e)
   {
    throw e;
   }
   finally
   {
    graphics.dispose();
   }
  }
 [description("控件選擇區(qū)域"),category("behavior")] 
  public  rectangle selectrectangle
  {
   get 
   {
    rectangle selectrectangler=new rectangle();
    selectrectangler.x = this.location.x+7;
    selectrectangler.y = this.location.y+7;
    selectrectangler.height = this.height-15;
    selectrectangler.width = this.width-15;
    return selectrectangler;
   }
  
  }
 
redrawcontrol中定義了rectangle (矩形),讓后填充改矩形的邊框:graphics.drawrectangle(borderpen, rectborder);,這里要說明的是邊框外面還有編輯框,所以大小不是控件的大小。drawselector就是繪制8個選擇框的,基本和繪制邊框差不多,即使定義好坐標(biāo)就可以了。干好了之后不要忘了釋放資源:graphics.dispose();
selectrectangle:控件所選擇的rectangle,最終要得就是它了
ok,這樣一個基本的東西出來了,下面我們要寫他的move和resize函數(shù)了,先添加事件:
this.resize += new system.eventhandler(this.shapeex_resize);
   this.mouseup += new system.windows.forms.mouseeventhandler(this.shapeex_mouseup);
   this.mousemove += new system.windows.forms.mouseeventhandler(this.shapeex_mousemove);
   this.mouseleave += new system.eventhandler(this.shapeex_mouseleave);
   this.mousedown += new system.windows.forms.mouseeventhandler(this.shapeex_mousedown);
原理:當(dāng)鼠標(biāo)點(diǎn)擊的時候this.mousedown,記錄鼠標(biāo)的位置,控件的原始位置和大小,判斷位置:_rectleftbottomselector.contains(e.x,e.y):表示點(diǎn)擊的是左下,設(shè)置鼠標(biāo),記錄狀態(tài)this._selectselctedindex:判斷點(diǎn)擊了那個選擇框,取值0-8:0代表移動整個控件,1是右上移動,2-8順時針?biāo)饕x擇框。this.mousemove處理如何改變控件的大小和位置
case 0://只移動位置
     this.location=new point(cursor.position.x-(_mouselocation.x-_selflocation.x),cursor.position.y-(_mouselocation.y-_selflocation.y));
     break;
1,5不僅移動位置,還改變大小,2,3,4,6,7,8只改變大小
其他則是清理工作。
好了,那么趕緊編譯。生成就可以了。

新聞熱點(diǎn)
疑難解答
圖片精選