Добавление блоков фильтров в заголовки столбцов ListView в C # и WinForms - PullRequest
3 голосов
/ 09 октября 2011

В проводнике Windows (по крайней мере, в Win7) при наведении указателя мыши на заголовок столбца появляется поле фильтра со стрелкой, которое позволяет фильтровать результаты в ListView, поэтому, например, вы можете отображать только файлы, начинающиеся с A "или файлы> 128 МБ. Можно ли включить эту функцию в базовом элементе управления ListView в C # без разделения на подклассы или изменения ListView?

Ответы [ 2 ]

4 голосов
/ 10 октября 2011

Вот код для игры.Добавьте новый класс в ваш проект и вставьте код, показанный ниже.Компиляция.Удалите новый элемент управления ListViewEx из верхней части панели инструментов на форму.В конструкторе формы вызовите метод SetHeaderDropdown (), чтобы включить кнопку.Реализуйте событие HeaderDropdown, чтобы вернуть элемент управления для отображения.Например:

public partial class Form1 : Form {
    public Form1() {
        InitializeComponent();
        listViewEx1.SetHeaderDropdown(0, true);
        listViewEx1.HeaderDropdown += listViewEx1_HeaderDropdown;
    }

    void listViewEx1_HeaderDropdown(object sender, ListViewEx.HeaderDropdownArgs e) {
        e.Control = new UserControl1();
    }
}

У приведенного ниже кода есть недостаток, всплывающее окно отображается в форме.Который не может быть слишком маленьким и отвлекает внимание от основной формы.Отметьте этот ответ о том, как реализовать элемент управления, который может отображаться в виде окна верхнего уровня без необходимости в форме.Код:

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Drawing;
using System.Windows.Forms;
using System.Runtime.InteropServices;

class ListViewEx : ListView {
    public class HeaderDropdownArgs : EventArgs {
        public int Column { get; set; }
        public Control Control { get; set; }
    }
    public event EventHandler<HeaderDropdownArgs> HeaderDropdown;

    public void SetHeaderDropdown(int column, bool enable) {
        if (column < 0 || column >= this.Columns.Count) throw new ArgumentOutOfRangeException("column");
        while (HeaderDropdowns.Count < this.Columns.Count) HeaderDropdowns.Add(false);
        HeaderDropdowns[column] = enable;
        if (this.IsHandleCreated) SetDropdown(column, enable);
    }
    protected void OnHeaderDropdown(int column) {
        var handler = HeaderDropdown;
        if (handler == null) return;
        var args = new HeaderDropdownArgs() { Column = column };
        handler(this, args);
        if (args.Control == null) return;
        var frm = new Form();
        frm.FormBorderStyle = FormBorderStyle.FixedSingle;
        frm.ShowInTaskbar = false;
        frm.ControlBox = false;
        args.Control.Location = Point.Empty;
        frm.Controls.Add(args.Control);
        frm.Load += delegate { frm.MinimumSize = new Size(1, 1);  frm.Size = frm.Controls[0].Size; };
        frm.Deactivate += delegate { frm.Dispose(); };
        frm.StartPosition = FormStartPosition.Manual;
        var rc = GetHeaderRect(column);
        frm.Location = this.PointToScreen(new Point(rc.Right - SystemInformation.MenuButtonSize.Width, rc.Bottom));
        frm.Show(this.FindForm());
    }

    protected override void OnHandleCreated(EventArgs e) {
        base.OnHandleCreated(e);
        if (this.Columns.Count == 0 || Environment.OSVersion.Version.Major < 6 || HeaderDropdowns == null) return;
        for (int col = 0; col < HeaderDropdowns.Count; ++col) {
            if (HeaderDropdowns[col]) SetDropdown(col, true);
        }
    }

    private Rectangle GetHeaderRect(int column) {
        IntPtr hHeader = SendMessage(this.Handle, LVM_GETHEADER, IntPtr.Zero, IntPtr.Zero);
        RECT rc;
        SendMessage(hHeader, HDM_GETITEMRECT, (IntPtr)column, out rc);
        return new Rectangle(rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top);
    }

    private void SetDropdown(int column, bool enable) {
        LVCOLUMN lvc = new LVCOLUMN();
        lvc.mask = LVCF_FMT;
        lvc.fmt = enable ? LVCFMT_SPLITBUTTON : 0;
        IntPtr res = SendMessage(this.Handle, LVM_SETCOLUMN, (IntPtr)column, ref lvc);
    }

    protected override void WndProc(ref Message m) {
        Console.WriteLine(m);
        if (m.Msg == WM_NOTIFY) {
            var hdr = (NMHDR)Marshal.PtrToStructure(m.LParam, typeof(NMHDR));
            if (hdr.code == LVN_COLUMNDROPDOWN) {
                var info = (NMLISTVIEW)Marshal.PtrToStructure(m.LParam, typeof(NMLISTVIEW));
                OnHeaderDropdown(info.iSubItem);
                return;
            }
        }
        base.WndProc(ref m);
    }

    private List<bool> HeaderDropdowns = new List<bool>();

    // Pinvoke
    [DllImport("user32.dll")]
    private static extern IntPtr SendMessage(IntPtr hWnd, int msg, IntPtr wp, IntPtr lp);
    [DllImport("user32.dll")]
    private static extern IntPtr SendMessage(IntPtr hWnd, int msg, IntPtr wp, ref LVCOLUMN lvc);
    [DllImport("user32.dll")]
    private static extern IntPtr SendMessage(IntPtr hWnd, int msg, IntPtr wp, out RECT rc);
    [DllImport("user32.dll")]
    private static extern IntPtr SetParent(IntPtr hWnd, IntPtr hParent);

    private const int LVM_SETCOLUMN = 0x1000 + 96;
    private const int LVCF_FMT = 1;
    private const int LVCFMT_SPLITBUTTON = 0x1000000;
    private const int WM_NOTIFY = 0x204e;
    private const int LVN_COLUMNDROPDOWN = -100 - 64;
    private const int LVM_GETHEADER = 0x1000 + 31;
    private const int HDM_GETITEMRECT = 0x1200 + 7;


    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
    private struct LVCOLUMN {
        public uint mask;
        public int fmt;
        public int cx;
        public string pszText;
        public int cchTextMax;
        public int iSubItem;
        public int iImage;
        public int iOrder;
        public int cxMin;
        public int cxDefault;
        public int cxIdeal;
    }

    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
    private struct POINT {
        public int x, y;
    }
    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
    private struct RECT {
        public int left, top, right, bottom; 
    }
    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
    private struct NMHDR {
        public IntPtr hwndFrom;
        public IntPtr idFrom;
        public int code;
    }
    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
    private struct NMLISTVIEW {
        public NMHDR hdr;
        public int iItem;
        public int iSubItem;
        public uint uNewState;
        public uint uOldState;
        public uint uChanged;
        public POINT ptAction;
        public IntPtr lParam;
    }
}
0 голосов
/ 09 октября 2011

Может быть сложно реализовать интерфейс того же типа, но вы можете заставить ваш ListView отвечать на содержимое TextBox, обрабатывая событие TextBox s TextChanged и фильтруя список на основесодержимое.Если вы поместите список в DataTable, тогда фильтрация будет легкой, и вы сможете каждый раз менять свой ListView.

Конечно, это зависит от того, сколько элементов в вашем списке.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...