2013-04-15 123 views
4

我有這樣的事件:當快速移動鼠標時,爲什麼新窗體不能粘住鼠標的移動位置?

protected override void OnMouseDown(MouseEventArgs e) 
{ 
    mOffset = new Point(Width/2 - e.X, Height/2 - e.Y); 
    mCurrentPoint = PointToScreen(new Point(e.X + mOffset.X, e.Y + mOffset.Y)); 
    mTargetPoint = mCurrentPoint; 
    mTimer.Enabled = true; 
} 

而這個事件:

protected override void OnMouseMove(MouseEventArgs e) 
{ 
    mTargetPoint = PointToScreen(new Point(e.X + mOffset.X, e.Y + mOffset.Y)); 
    mTimer.Enabled = true; 
} 

如果我只是移動鼠標快我失去焦點的形式,我不動,形式留在只放置鼠標移動。

但如果我點擊鼠標按下按鈕,然後移動鼠標,快捷的形式是焦點一路。

所以我試圖兩線從OnMouseDown事件複製到OnMouseMove事件:

mOffset = new Point(Width/2 - e.X, Height/2 - e.Y); 
mTargetPoint = mCurrentPoint; 

但是,一旦我提出這兩條線Move事件沒有發生。不管我做什麼,表格都不會移動。

如果它的需要,那麼這是完整的表單代碼:

using System; 
using System.Collections.Generic; 
using System.ComponentModel; 
using System.Data; 
using System.Drawing; 
using System.Text; 
using System.Windows.Forms; 
using System.Drawing.Drawing2D; 
using System.IO; 
using System.Drawing.Imaging; 

namespace Magnifier20070401 
{ 
    public partial class MagnifierForm : Form 
    { 
     public MagnifierForm()//Configuration configuration, Point startPoint) 
     { 
      InitializeComponent(); 

      //--- My Init --- 
      //mConfiguration = configuration; 
      FormBorderStyle = FormBorderStyle.None; 

      ShowInTaskbar = false;//mConfiguration.ShowInTaskbar; 
      TopMost = true;//mConfiguration.TopMostWindow; 
      Width = 150;// mConfiguration.MagnifierWidth; 
      Height = 150;// mConfiguration.MagnifierHeight; 

      // Make the window (the form) circular 
      GraphicsPath gp = new GraphicsPath(); 
      gp.AddEllipse(ClientRectangle); 
      Region = new Region(gp); 

      mImageMagnifier = ScreenVideoRecorder.Properties.Resources.magnifierGlass; 

      mTimer = new Timer(); 
      mTimer.Enabled = true; 
      mTimer.Interval = 20; 
      mTimer.Tick += new EventHandler(HandleTimer); 

      mScreenImage = new Bitmap(Screen.PrimaryScreen.Bounds.Width, 
            Screen.PrimaryScreen.Bounds.Height); 

      mStartPoint = new Point(500, 500);// startPoint; 
      mTargetPoint = new Point(500, 500); // startPoint; 

      /*if (mConfiguration.ShowInTaskbar) 
       ShowInTaskbar = true; 
      else 
       ShowInTaskbar = false;*/ 
     } 

     protected override void OnShown(EventArgs e) 
     { 
      RepositionAndShow(); 
     } 

     private delegate void RepositionAndShowDelegate(); 

     private void RepositionAndShow() 
     { 
      if (InvokeRequired) 
      { 
       Invoke(new RepositionAndShowDelegate(RepositionAndShow)); 
      } 
      else 
      { 
       // Capture the screen image now! 
       Graphics g = Graphics.FromImage(mScreenImage); 
       g.CopyFromScreen(0, 0, 0, 0, new Size(mScreenImage.Width, mScreenImage.Height)); 
       g.Dispose();     

       //if (mConfiguration.HideMouseCursor) 
        // Cursor.Hide(); 
       //else 
        Cursor = Cursors.Cross; 

       Capture = true; 

       /*if (mConfiguration.RememberLastPoint) 
       { 
        mCurrentPoint = mLastMagnifierPosition; 
        Cursor.Position = mLastMagnifierPosition; 
        Left = (int)mCurrentPoint.X - Width/2; 
        Top = (int)mCurrentPoint.Y - Height/2; 
       } 
       else 
       {*/ 
        mCurrentPoint = Cursor.Position; 
       //} 
       Show(); 
      } 
     } 

     void HandleTimer(object sender, EventArgs e) 
     { 
      float dx = /*mConfiguration.SpeedFactor*/ (float)0.35 * (mTargetPoint.X - mCurrentPoint.X); 
      float dy = /*mConfiguration.SpeedFactor*/ (float)0.35 * (mTargetPoint.Y - mCurrentPoint.Y); 

      if (mFirstTime) 
      { 
       mFirstTime = false; 

       mCurrentPoint.X = mTargetPoint.X; 
       mCurrentPoint.Y = mTargetPoint.Y; 

       Left = (int)mCurrentPoint.X - Width/2; 
       Top = (int)mCurrentPoint.Y - Height/2; 

       return; 
      } 

      mCurrentPoint.X += dx; 
      mCurrentPoint.Y += dy; 

      if (Math.Abs(dx) < 1 && Math.Abs(dy) < 1) 
      { 
       mTimer.Enabled = false; 
      } 
      else 
      { 
       // Update location 
       Left = (int)mCurrentPoint.X - Width/2; 
       Top = (int)mCurrentPoint.Y - Height/2; 
       mLastMagnifierPosition = new Point((int)mCurrentPoint.X, (int)mCurrentPoint.Y); 
      } 

      Refresh(); 
     } 


     protected override void OnMouseDown(MouseEventArgs e) 
     { 
      mOffset = new Point(Width/2 - e.X, Height/2 - e.Y); 
      mCurrentPoint = PointToScreen(new Point(e.X + mOffset.X, e.Y + mOffset.Y)); 
      mTargetPoint = mCurrentPoint; 
      mTimer.Enabled = true; 
     } 

     protected override void OnMouseUp(MouseEventArgs e) 
     { 
      //if (mConfiguration.CloseOnMouseUp) 
      //{ 
      /* Close(); 
       mScreenImage.Dispose(); 
      //} 

      Cursor.Show(); 
      Cursor.Position = mStartPoint; */   
     } 

     protected override void OnMouseMove(MouseEventArgs e) 
     { 
      //if (e.Button == MouseButtons.Left) 
      //{ 
       mOffset = new Point(Width/2 - e.X, Height/2 - e.Y); 
       mTargetPoint = PointToScreen(new Point(e.X + mOffset.X, e.Y + mOffset.Y)); 
       //mTargetPoint = mCurrentPoint; 
       mTimer.Enabled = true; 
      //} 
     } 

     protected override void OnPaintBackground(PaintEventArgs e) 
     { 
      /*if (mConfiguration.DoubleBuffered) 
      { 
       // Do not paint background (required for double buffering)! 
      } 
      else 
      { 
       base.OnPaintBackground(e); 
      }*/ 
      base.OnPaintBackground(e); 
     } 

     protected override void OnPaint(PaintEventArgs e) 
     { 
      if (mBufferImage == null) 
      { 
       mBufferImage = new Bitmap(Width, Height); 
      } 
      Graphics bufferGrf = Graphics.FromImage(mBufferImage); 

      Graphics g; 

      /*if (mConfiguration.DoubleBuffered) 
      { 
       g = bufferGrf; 
      } 
      else 
      {*/ 
       g = e.Graphics; 
      //} 

      if (mScreenImage != null) 
      { 
       Rectangle dest = new Rectangle(0, 0, Width, Height); 
       int w = (int)(Width/3.0);//mConfiguration.ZoomFactor); 
       int h = (int)(Height/3.0);//mConfiguration.ZoomFactor); 
       int x = Left - w/2 + Width/2; 
       int y = Top - h/2 + Height/2; 

       g.DrawImage(
        mScreenImage, 
        dest, 
        x, y, 
        w, h, 
        GraphicsUnit.Pixel); 
      } 

      if (mImageMagnifier != null) 
      { 
       g.DrawImage(mImageMagnifier, 0, 0, Width, Height); 
      } 

      //if (mConfiguration.DoubleBuffered) 
      //{ 
       e.Graphics.DrawImage(mBufferImage, 0, 0, Width, Height); 
      //}  
     } 


     //--- Data Members --- 
     #region Data Members 
     private Timer mTimer; 
     private Configuration mConfiguration; 
     private Image mImageMagnifier; 
     private Image mBufferImage = null; 
     private Image mScreenImage = null; 
     private Point mStartPoint; 
     private PointF mTargetPoint; 
     private PointF mCurrentPoint; 
     private Point mOffset; 
     private bool mFirstTime = true; 
     private static Point mLastMagnifierPosition = Cursor.Position; 
     #endregion 

     private void MagnifierForm_KeyDown(object sender, KeyEventArgs e) 
     { 
      if (e.Control && e.KeyCode.ToString() == "M") 
      { 
       this.Close(); 
      } 
     } 
    } 
} 

在Form1我只是做一個新實例的形式,然後我可以移動它。 爲什麼當鼠標不按鍵時快速移動鼠標會失去焦點?

回答

2

它是不是一個事件,它是一種方法。並且你沒有得到默認行爲(捕獲鼠標),因爲你忘記調用base.OnMouseDown()。在MSDN Library嚴詞警告一下:

給繼承:

在派生類中重寫onmousedown事件, 一定要調用基類的onmousedown事件方法,以便註冊 委託接收該事件。

好,和其他的東西,基類的方法一樣。就像捕獲鼠標一樣。當你真的知道你在做什麼時,只跳過調用基類方法。很難弄清楚,你真的需要知道基類方法的作用。它始終只是像發射事件一樣簡單,不是而是。通常,類越複雜,它越有可能在基本方法中拋棄一些代碼。 DataGridView是松鼠之王。您可以從參考源獲取該信息。

只需在該方法的底部添加此行來解決問題:

base.OnMouseDown(e); 

,做在的OnMouseMove()重寫相同。不是因爲這是解決這個特定問題的必要條件,因爲文檔說你應該。

2

在你的MouseDown你應該把this.Capture = true;,並在您的MouseUp把this.Capture.false;

如果快速移動鼠標它可能移動窗外,默認情況下只鼠標是在將得到鼠標的窗口消息。使用Control.Capture修復了這個問題。

相關問題