Пожалуйста, опишите ваш код.

могу получить событие колесика мыши в главной форме.

В качестве демонстрации я привел простой пример:

public partial class Form1 : Form
{
    public Form1()
    {
        InitializeComponent();

        this.panel1.MouseWheel += new MouseEventHandler(panel1_MouseWheel);
        this.panel1.MouseMove += new MouseEventHandler(panel1_MouseWheel);

        Form2 f2 = new Form2();
        f2.Show(this);
    }

    private void panel1_MouseWheel(object sender, MouseEventArgs e)
    {
        if(e.Delta != 0)
        Console.Out.WriteLine(e.Delta);
    }
}

public partial class Form2 : Form
{
    public Form2()
    {
        InitializeComponent();

        this.MouseMove += new MouseEventHandler(Form2_MouseMove);
        this.MouseWheel += new MouseEventHandler(Form2_MouseMove);
    }

    private void Form2_MouseMove(object sender, MouseEventArgs e)
    {
        if(e.Delta != 0)
            Console.Out.WriteLine(e.Delta);
    }
}

Я получаю событие колеса мыши в Form2, но не в Form1, есть идеи?

Ура,

Джеймс

 Qwertie12 июн. 2017 г., 00:05
Вечный недостаток колеса мыши в том, что Microsoft решила рассматривать его больше как событие клавиатуры, чем как событие мыши, поэтому сообщения колеса мыши попадают в элемент управления с фокусом клавиатуры, заставляя почти каждое приложение, использующее колесо мыши, выполнять некоторые действия. своего рода обходной путь.

Ответы на вопрос(8)

я написал простой универсальный класс, чтобы легко решить эту проблему:

using System;
using System.Windows.Forms;
using System.Runtime.InteropServices;
using System.Drawing;

namespace MyNamespace
{
  public class MouseWheelManagedForm : Form, IMessageFilter
  {
    private bool managed;

    public MouseWheelManagedForm () : this (true) {
   }

    public MouseWheelManagedForm (bool start) {
      managed = false;
      if (start)
        ManagedMouseWheelStart();
    }

    protected override void Dispose (bool disposing) {
      if (disposing)
        ManagedMouseWheelStop();
      base.Dispose(disposing);
    }

    /************************************
     * IMessageFilter implementation
     * *********************************/
    private const int WM_MOUSEWHEEL = 0x20a;
    // P/Invoke declarations
    [DllImport("user32.dll")]
    private static extern IntPtr WindowFromPoint (Point pt);
    [DllImport("user32.dll")]
    private static extern IntPtr SendMessage (IntPtr hWnd, int msg, IntPtr wp, IntPtr lp);

    private bool IsChild (Control ctrl) {
      Control loopCtrl = ctrl;

      while (loopCtrl != null && loopCtrl != this)
        loopCtrl = loopCtrl.Parent;

      return (loopCtrl == this);
    }

    public bool PreFilterMessage (ref Message m) {
      if (m.Msg == WM_MOUSEWHEEL) {
        //Ensure the message was sent to a child of the current form
        if (IsChild(Control.FromHandle(m.HWnd))) {
          // Find the control at screen position m.LParam
          Point pos = new Point(m.LParam.ToInt32() & 0xffff, m.LParam.ToInt32() >> 16);

          //Ensure control under the mouse is valid and is not the target control
          //otherwise we'd be trap in a loop.
          IntPtr hWnd = WindowFromPoint(pos);
          if (hWnd != IntPtr.Zero && hWnd != m.HWnd && Control.FromHandle(hWnd) != null) {
            SendMessage(hWnd, m.Msg, m.WParam, m.LParam);
            return true;
          }
        }
      }
      return false;
    }

    /****************************************
     * MouseWheelManagedForm specific methods
     * **************************************/
    public void ManagedMouseWheelStart () {
      if (!managed) {
        managed = true;
        Application.AddMessageFilter(this);
      }
    }

    public void ManagedMouseWheelStop () {
      if (managed) {
        managed = false;
        Application.RemoveMessageFilter(this);
      }
    }

  }
}

Оттуда вам нужно только наследовать форму от этого класса вместо формы для каждой формы, для которой вам нужно управлять мышью:

using System;
using System.Collections.Generic;
using System.Data;
using System.Linq;
using System.Windows.Forms;

namespace MyApp
{
  public partial class MyForm : MyNamespace.MouseWheelManagedForm
  {
    public MyForm ()
    {
      InitializeComponent();
    }

  }
}

Надеюсь, это поможет кому-то еще (кроме меня).

 colin lamarre21 янв. 2015 г., 14:51
Кстати, вы можете использовать это. Содержит вместо IsChild.

что форма1 имеет фокус, а не панель1. ... который, конечно, означает, что будут запускаться события form1, а не панели1.

Я пересоздал ваш сценарий со следующими изменениями в конструкторе в Form1 и убедился, что он запускает событие колеса прокрутки.

public Form1()
{
        InitializeComponent(); 

        /*  --- Old code that don't work ---
            this.panel1.MouseWheel += new MouseEventHandler(panel1_MouseWheel);
            this.panel1.MouseMove += new MouseEventHandler(panel1_MouseWheel);
        */

        this.MouseWheel += new MouseEventHandler(panel1_MouseWheel);
        this.MouseMove += new MouseEventHandler(panel1_MouseWheel);

        Form2 f2 = new Form2();
        f2.Show(this);
    }
}

вы немного неправильно поняли мою проблему. В моей основной форме с кодом ниже я не получаю событие MouseWheel:

public Form1()
    {
        InitializeComponent();

        this.panel1.MouseWheel += new MouseEventHandler(panel1_MouseWheel);
        this.panel1.MouseMove += new MouseEventHandler(panel1_MouseWheel);    
    }

Но я получаю событие с:

public Form1()
    {
        InitializeComponent();
        this.MouseWheel += new MouseEventHandler(panel1_MouseWheel);

    }

Надеюсь, это будет понятнее без путаницы в Form2. Я просто пытаюсь получить события MouseWheel на панели в моей основной форме.

Ура,

Джеймс

 ToolmakerSteve14 окт. 2017 г., 21:49
Кстати, это должно быть редактирование вопроса, а не ответ.

фокус может быть только у элемента, размещенного внутри панели. Панель будет получать событие MouseWheel только после того, как что-то будет помещено в нее, и эта вещь будет в фокусе. Простое наведение на панель и перемещение колесика мыши отправит событие в форму, а не на панель.

В этом разница между вашими двумя примерами.

 Jerry26 янв. 2009 г., 18:36
Не правда. В приложении, которое я пишу, у меня есть панель, которая является моим графическим пространством отображения. Хотя вы не можете щелкнуть фокус на панели, вы можете сказать (в коде) <code> myPanel.Focus () </ code>, чтобы панель отвлекала внимание от всего остального, чтобы мой код дисплея работал ,
 Sam Meldrum27 янв. 2009 г., 11:02
@ Джерри - я проверил это, и мой тест показывает, что я прав. Если у вас ничего нет на панели, даже вызов myPanel.Focus () не изменит направление событий MouseWheel. Я могу получить события MouseWheel только тогда, когда что-то на панели имеет фокус.
this.MouseWheel += pictureBox1_MouseWheel; //tanımlama
void pictureBox1_MouseWheel(object sender, MouseEventArgs e)
            {
                if (Convert.ToString(e.Delta) == "120")
                {
                    //yukarı
                }
                else if (Convert.ToString(e.Delta) == "-120")
                {
                    //aşağı
                }
            }
 Vitalii Vitrenko14 авг. 2017 г., 23:07
Пожалуйста, опишите ваш код.

Может быть, это будет работать для вас?

public partial class Form1 : Form {
    public Form1() {
        InitializeComponent();
        Form2 f2 = new Form2();
        f2.MouseWheel += new MouseEventHandler(panel1_MouseWheel);
        f2.MouseMove += new MouseEventHandler(panel1_MouseWheel);
        f2.Show(this);
    }

    private void panel1_MouseWheel(object sender, MouseEventArgs e)
    {
        if(e.Delta != 0) Console.Out.WriteLine(e.Delta);
    }
}

Добавить другое событие панелиMouseEnter и в его функции обратного вызова получить фокус ввода:

void MouseEnterEvent()
{
   this.Panel.Focus();
}

что OP хочет получать события прокрутки, когда над панелью находится указатель мыши, даже если панель не имеет фокуса.

Способ достижения этого поведения объясняется здесь:

http://social.msdn.microsoft.com/forums/en-US/winforms/thread/eb922ed2-1036-41ca-bd15-49daed7b637c/

и здесь:

http://social.msdn.microsoft.com/forums/en-US/winforms/thread/6bfb9287-986d-4c60-bbcc-23486e239384/

Один из фрагментов кода, взятых из связанного форума:

using System;
using System.ComponentModel;
using System.Drawing;
using System.Windows.Forms;
using System.Runtime.InteropServices;

namespace WindowsApplication1 {
  public partial class Form1 : Form, IMessageFilter {
    public Form1() {
      InitializeComponent();
      Application.AddMessageFilter(this);
    }

    public bool PreFilterMessage(ref Message m) {
      if (m.Msg == 0x20a) {
        // WM_MOUSEWHEEL, find the control at screen position m.LParam
        Point pos = new Point(m.LParam.ToInt32() & 0xffff, m.LParam.ToInt32() >> 16);
        IntPtr hWnd = WindowFromPoint(pos);
        if (hWnd != IntPtr.Zero && hWnd != m.HWnd && Control.FromHandle(hWnd) != null) {
          SendMessage(hWnd, m.Msg, m.WParam, m.LParam);
          return true;
        }
      }
      return false;
    }

    // P/Invoke declarations
    [DllImport("user32.dll")]
    private static extern IntPtr WindowFromPoint(Point pt);
    [DllImport("user32.dll")]
    private static extern IntPtr SendMessage(IntPtr hWnd, int msg, IntPtr wp, IntPtr lp);
  }
}

Этот код будет в основном перехватывать все события wm_mousewheel и перенаправлять их на элемент управления, над которым в данный момент наведена мышь. На панели больше не нужно фокусироваться, чтобы получать события колеса.

 Bryce Wagner29 мая 2015 г., 21:03
Может быть, некоторые из этих приведений не нужны :-) Но это должно предотвратить любые исключения приведения. (int) (long) будет понижен с 64-битного IntPtr до int, (uint) заставит его выполнять сдвиги без знака (вместо сдвигов со знаком), (ushort), чтобы извлечь младшие 16 бит, и (short), чтобы преобразовать его в подписанное значение.
 Bryce Wagner28 окт. 2015 г., 19:16
@glancep Извините, может быть, я не понял, где происходило исключение. Это происходит только в 64-битном режиме. Если у вас есть 64-разрядный IntPtr со значениями от int.MinValue до int.MaxValue, IntPtr.ToInt32 () вызовет исключение переполнения.
 Bryce Wagner29 мая 2015 г., 21:00
"m.LParam.ToInt32 () & 0xffff" должен быть "(короткий) (ushort) (uint) (int) (длинный) m.LParam" и "m.LParam.ToInt32 () >> 16" должен быть "( short) (ushort) (((uint) (int) (long) m.LParam) >> 16) "
 Martin Bliss07 авг. 2012 г., 02:44
Это должно быть отмечено ответ. Он работает во всех сценариях и может быть легко изменен (как я сделал для своих личных нужд).
 Bryce Wagner28 окт. 2015 г., 19:18
Тем не менее, это лучший конструктор. new Point ((int) (long) m.LParam) определенно намного чище, чем бардак, который я выложил выше.

Ваш ответ на вопрос