Кнопка Windows.Forms с выпадающим меню

Я разрабатываю простое приложение на C # с использованием Windows.Forms в .NET. Мне нужна кнопка, которая покажет выпадающее меню с подкатегориями - очень похоже на ToolStripMenu, но кнопка, вы знаете. Я искал это и не мог найти никаких вариантов.

Мой вопрос: есть ли способ сделать это, может быть, какое-то свойство секретной кнопки, которое позволяет прикрепить к нему меню?

Любая помощь будет оценена.

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

Легко было, мы можем это сделать. это может помочь :)

ContextMenuStrip contextMenuStrip1 = new ContextMenuStrip();

        private void button1_Click(object sender, EventArgs e)
        {
            contextMenuStrip1.Items.Clear();
            contextMenuStrip1.Items.Add("item1");
            contextMenuStrip1.Items.Add("item2");

            contextMenuStrip1.Show(button1, new Point(0, button1.Height));
        }

        private void contextMenuStrip1_ItemClicked(object sender, ToolStripItemClickedEventArgs e)
        {
            if (e.ClickedItem.Text == "item1")
            {
                MessageBox.Show(e.ClickedItem.Text);
            }
        }
Решение Вопроса

Вы можете показать ContextMenuStrip для события click:

private void button1_Click(object sender, EventArgs e) {
  contextMenuStrip1.Show(button1, new Point(0, button1.Height));
}

Чтобы самостоятельно определить, показывать ли меню над или под кнопкой, вы можете попробовать использовать этот код, который измеряет меню и определяет, будет ли оно частично за кадром:

private void button1_Click(object sender, EventArgs e) {
  Point screenPoint = button1.PointToScreen(new Point(button1.Left, button1.Bottom));
  if (screenPoint.Y + contextMenuStrip1.Size.Height > Screen.PrimaryScreen.WorkingArea.Height) {
    contextMenuStrip1.Show(button1, new Point(0, -contextMenuStrip1.Size.Height));
  } else {
    contextMenuStrip1.Show(button1, new Point(0, button1.Height));
  }    
}
 tonytony29 мая 2012 г., 19:02
Как насчет перекрытия кнопки? Я думаю, что я должен знать, пойдет ли это вверх или вниз.
 tonytony29 мая 2012 г., 19:12
Большое спасибо :)
 29 мая 2012 г., 18:59
@tonytony Это должно произойти автоматически. Если меню расположено ниже нижней части экрана, окна будут подталкивать его вверх, чтобы оно было все видимым.
 29 мая 2012 г., 19:08
@tonytony Обновленный пример.
 tonytony29 мая 2012 г., 18:55
Похоже, что мне нужно. Знаете ли вы хороший способ определить, должно ли меню быть выпадающим из нижней части кнопки или выпадающим из верхней границы кнопок?

расширяющийся@Jaex ответ немного, чтобы учесть разделительную линию, условное рисование стрелки, если ничего не настроено, и отдельное событие щелчка для основного тела кнопки и стрелки меню.

Следует отметить, что для лучшего выравнивания вы можете установитьbutton.TextAlign = System.Drawing.ContentAlignment.MiddleLeft;

enter image description here

Вот мое небольшое улучшение

public class SplitButton : Button
{
    [DefaultValue(null), Browsable(true),
    DesignerSerializationVisibility(DesignerSerializationVisibility.Visible)]
    public ContextMenuStrip Menu { get; set; }

    [DefaultValue(20), Browsable(true),
    DesignerSerializationVisibility(DesignerSerializationVisibility.Visible)]
    public int SplitWidth { get; set; }

    public SplitButton() 
    {
        SplitWidth = 20;
    }

    protected override void OnMouseDown(MouseEventArgs mevent)
    {
        var splitRect = new Rectangle(this.Width - this.SplitWidth, 0, this.SplitWidth, this.Height);

        // Figure out if the button click was on the button itself or the menu split
        if (Menu != null && 
            mevent.Button == MouseButtons.Left &&
            splitRect.Contains(mevent.Location) )
        {
            Menu.Show(this, 0, this.Height);    // Shows menu under button
            //Menu.Show(this, mevent.Location); // Shows menu at click location
        }
        else
        {
            base.OnMouseDown(mevent);
        }
    }

    protected override void OnPaint(PaintEventArgs pevent)
    {
        base.OnPaint(pevent);

        if (this.Menu != null && this.SplitWidth > 0)
        { 
            // Draw the arrow glyph on the right side of the button
            int arrowX = ClientRectangle.Width - 14;
            int arrowY = ClientRectangle.Height / 2 - 1;

            var arrowBrush = Enabled ? SystemBrushes.ControlText : SystemBrushes.ButtonShadow;
            var arrows = new[] { new Point(arrowX, arrowY), new Point(arrowX + 7, arrowY), new Point(arrowX + 3, arrowY + 4) };
            pevent.Graphics.FillPolygon(arrowBrush, arrows);

            // Draw a dashed separator on the left of the arrow
            int lineX = ClientRectangle.Width - this.SplitWidth;
            int lineYFrom = arrowY - 4;
            int lineYTo = arrowY + 8;
            using( var separatorPen = new Pen(Brushes.DarkGray){DashStyle = DashStyle.Dot})
            {
                pevent.Graphics.DrawLine(separatorPen, lineX, lineYFrom, lineX, lineYTo);
            }
        }
    }
}
 14 февр. 2019 г., 09:34
Как вызвать выпадающее меню, не используя мышь, используя только клавиатуру?
 21 сент. 2018 г., 00:29
Это очень хорошо @Sverrir Я искал этот тип управления в своих проектах
 26 февр. 2017 г., 22:18
Джеки прав. Раскрывающийся список & quot; щелкните область & quot; не охватывает всю высоту кнопки, если этот параметр не равен 0.
 01 дек. 2017 г., 17:11
Я отредактировал код, чтобы исправить ошибку, обнаруженную @Jacky.
 11 янв. 2015 г., 03:29
var splitRect = new Rectangle(this.Width - this.SplitWidth, this.Location.Y, this.SplitWidth, this.Height); второй параметр должен быть изменен на0

Показывать контекстное меню под кнопкой при ее нажатии.

Кнопка имеет стрелку вниз справа от нее, и вы можете выбрать меню из конструктора:

ss

С ShowMenuUnderCursor:

ss

Класс MenuButton:

public class MenuButton : Button
{
    [DefaultValue(null)]
    public ContextMenuStrip Menu { get; set; }

    [DefaultValue(false)]
    public bool ShowMenuUnderCursor { get; set; }

    protected override void OnMouseDown(MouseEventArgs mevent)
    {
        base.OnMouseDown(mevent);

        if (Menu != null && mevent.Button == MouseButtons.Left)
        {
            Point menuLocation;

            if (ShowMenuUnderCursor)
            {
                menuLocation = mevent.Location;
            }
            else
            {
                menuLocation = new Point(0, Height);
            }

            Menu.Show(this, menuLocation);
        }
    }

    protected override void OnPaint(PaintEventArgs pevent)
    {
        base.OnPaint(pevent);

        if (Menu != null)
        {
            int arrowX = ClientRectangle.Width - 14;
            int arrowY = ClientRectangle.Height / 2 - 1;

            Brush brush = Enabled ? SystemBrushes.ControlText : SystemBrushes.ControlDark;
            Point[] arrows = new Point[] { new Point(arrowX, arrowY), new Point(arrowX + 7, arrowY), new Point(arrowX + 3, arrowY + 4) };
            pevent.Graphics.FillPolygon(brush, arrows);
        }
    }
}
 27 нояб. 2014 г., 15:50
Фантастика! Большое спасибо за этот ответ
 tonytony07 июн. 2014 г., 00:00
Спасибо за ответ! Я действительно подумал о кнопке со стрелкой, когда разместил вопрос, и это отличная идея нарисовать стрелку с помощью Graphics API.
 02 нояб. 2015 г., 06:16
ЗачемMenuзначение всегдаnullнезависимо от того, как установитьContextMenuStrip стоимость имущества?
 31 мая 2016 г., 18:32
@Alex Добавьте новый объект ContextMenuStrip кMenu собственность, а неContextMenuStrip имущество.myButton.Menu = New ContextMenuStrip .. вы, вероятно, делаетеmyButton.ContextMenuStrip = New ContextMenuStrip
 24 дек. 2016 г., 02:49
Если вам просто захочется использовать кнопкуOnPaint событие вместо наследования отButtonне забудьте использовать размеры кнопки, а неClientRectangle& APOS; с. Как это:Button btnSender = (Button)sender; int arrowX = btnSender.Width - 14; int arrowY = btnSender.Height / 2 - 1;

Простейшим вариантом будет использование ToolStripDropDownButton в открепленной ToolStrip, которая показывает только одну кнопку. Затем вы можете добавить к нему подпункты и т. Д. Для этого: - перетащите Toolstrip на свой элемент управления / форму - используйте помощник по макету, чтобы добавить DropDownButton - установите для GripStyle значение Hidden - установить Dock на None

Результатом является отдельная кнопка в стиле панели инструментов, которая поддерживает раскрывающееся поведение, которое вы описали.

 tonytony29 мая 2012 г., 18:54
Спасибо за ответ, это чисто и понятно, но не совсем то, что мне нужно. Кнопка меню вроде не скучная, а я хочу что-нибудь более мясистое :)

Класс Jaex MenuButton выше был идеальным для меня. Я добавил логику ниже в OnMouseDown, чтобы контекстное меню отображалось только при нажатии на стрелку. Нормальное событие щелчка будет вызвано, если я нажму в большей части. Разрешено для & quot; По умолчанию & quot; нажмите действие.

if (Menu != null && mevent.Button == MouseButtons.Left)
{
    if (mevent.Location.X >= this.Width - 14)
    {
        System.Drawing.Point menuLocation;

        if (ShowMenuUnderCursor)
        {
            menuLocation = mevent.Location; 
        }
        else
        {
            menuLocation = new System.Drawing.Point(0, Height);
        }

        Menu.Show(this, menuLocation);
    }
}

Думал, что это может быть полезно для кого-то. Спасибо Jaex

Я тоже возился с этой проблемой и нашел чрезвычайно простое решение (хотя и немного грязное):ComboBox подButton, так что он показывает стрелку раскрывающегося списка прямо рядом с кнопкой.

Тогда используйтеSelectedIndexChanged изComboBox изменитьButton поведение, или делать то, что вы хотите, чтобы сделать это немедленно.

 tonytony12 апр. 2017 г., 17:23
Интересное решение! Я боюсь, что поведение при фокусировке вкладок выдаст вас.
 tonytony12 апр. 2017 г., 17:28
может быть. Если честно, я давно забросил windows.foms и c #.
 12 апр. 2017 г., 17:25
Просто установитеTabStop вFalse наComboBox и поймайте клавишу со стрелкой вниз наButton ?

Инфраструктура имеет WinDropDownButton:http://help.infragistics.com/Help/NetAdvantage/WinForms/2012.1/CLR2.0/html/WinDropDownButton_About_WinDropDownButton.html

Так что, безусловно, существует, однако вы можете не искать платного стороннего контроля.

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