// OpenTween - Client of Twitter // Copyright (c) 2007-2011 kiri_feather (@kiri_feather) // (c) 2008-2011 Moz (@syo68k) // (c) 2008-2011 takeshik (@takeshik) // (c) 2010-2011 anis774 (@anis774) // (c) 2010-2011 fantasticswallow (@f_swallow) // (c) 2011 kim_upsilon (@kim_upsilon) // All rights reserved. // // This file is part of OpenTween. // // This program is free software; you can redistribute it and/or modify it // under the terms of the GNU General Public License as published by the Free // Software Foundation; either version 3 of the License, or (at your option) // any later version. // // This program is distributed in the hope that it will be useful, but // WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY // or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License // for more details. // // You should have received a copy of the GNU General Public License along // with this program. If not, see , or write to // the Free Software Foundation, Inc., 51 Franklin Street - Fifth Floor, // Boston, MA 02110-1301, USA. #nullable enable using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Windows.Forms; using System.Drawing; using System.ComponentModel; using System.Runtime.InteropServices; using System.Diagnostics; namespace OpenTween.OpenTweenCustomControl { public sealed class DetailsListView : ListView { private Rectangle changeBounds; public ContextMenuStrip? ColumnHeaderContextMenuStrip { get; set; } public event EventHandler? VScrolled; public event EventHandler? HScrolled; public DetailsListView() { View = View.Details; FullRowSelect = true; HideSelection = false; DoubleBuffered = true; } /// /// 複数選択時の起点になるアイテム (selection mark) の位置を取得・設定する /// /// /// Items[idx].Selected の設定では mark が設定されるが、SelectedIndices.Add(idx) では設定されないため、 /// 主に後者と合わせて使用する /// public int SelectionMark { get => NativeMethods.ListView_GetSelectionMark(this.Handle); set => NativeMethods.ListView_SetSelectionMark(this.Handle, value); } public void SelectItems(int[] indices) { foreach (var index in indices) { if (index < 0 || index >= this.VirtualListSize) throw new ArgumentOutOfRangeException(nameof(indices)); NativeMethods.SelectItem(this, index); } this.OnSelectedIndexChanged(EventArgs.Empty); } public void SelectAllItems() { NativeMethods.SelectAllItems(this); this.OnSelectedIndexChanged(EventArgs.Empty); } public void ChangeItemBackColor(ListViewItem item, Color backColor) { if (item.BackColor == backColor) return; item.BackColor = backColor; this.RefreshItemBounds(item); } public void ChangeItemForeColor(ListViewItem item, Color foreColor) { if (item.ForeColor == foreColor) return; item.ForeColor = foreColor; this.RefreshItemBounds(item); } public void ChangeItemFontAndColor(ListViewItem item, Color foreColor, Font fnt) { if (item.ForeColor == foreColor && item.Font == fnt) return; item.ForeColor = foreColor; item.Font = fnt; this.RefreshItemBounds(item); } private void RefreshItemBounds(ListViewItem item) { try { var itemBounds = item.Bounds; var drawBounds = Rectangle.Intersect(this.ClientRectangle, itemBounds); if (drawBounds == Rectangle.Empty) return; this.changeBounds = drawBounds; this.Update(); this.changeBounds = Rectangle.Empty; } catch (ArgumentException) { //タイミングによりBoundsプロパティが取れない? this.changeBounds = Rectangle.Empty; } } [StructLayout(LayoutKind.Sequential)] private struct NMHDR { public IntPtr hwndFrom; public IntPtr idFrom; public int code; } [DebuggerStepThrough] protected override void WndProc(ref Message m) { const int WM_ERASEBKGND = 0x14; const int WM_PAINT = 0xF; const int WM_MOUSEWHEEL = 0x20A; const int WM_MOUSEHWHEEL = 0x20E; const int WM_HSCROLL = 0x114; const int WM_VSCROLL = 0x115; const int WM_KEYDOWN = 0x100; const int WM_USER = 0x400; const int WM_REFLECT = WM_USER + 0x1C00; const int WM_NOTIFY = 0x004E; const int WM_CONTEXTMENU = 0x7B; const int LVM_SETITEMCOUNT = 0x102F; const int LVN_ODSTATECHANGED = ((0 - 100) - 15); const long LVSICF_NOSCROLL = 0x2; const long LVSICF_NOINVALIDATEALL = 0x1; var hPos = -1; var vPos = -1; switch (m.Msg) { case WM_ERASEBKGND: if (this.changeBounds != Rectangle.Empty) m.Msg = 0; break; case WM_PAINT: if (this.changeBounds != Rectangle.Empty) { NativeMethods.ValidateRect(this.Handle, IntPtr.Zero); this.Invalidate(this.changeBounds); this.changeBounds = Rectangle.Empty; } break; case WM_HSCROLL: HScrolled?.Invoke(this, EventArgs.Empty); break; case WM_VSCROLL: VScrolled?.Invoke(this, EventArgs.Empty); break; case WM_MOUSEWHEEL: case WM_MOUSEHWHEEL: case WM_KEYDOWN: vPos = NativeMethods.GetScrollPosition(this, NativeMethods.ScrollBarDirection.SB_VERT); hPos = NativeMethods.GetScrollPosition(this, NativeMethods.ScrollBarDirection.SB_HORZ); break; case WM_CONTEXTMENU: if (m.WParam != this.Handle) { //カラムヘッダメニューを表示 this.ColumnHeaderContextMenuStrip?.Show(new Point(m.LParam.ToInt32())); return; } break; case LVM_SETITEMCOUNT: m.LParam = new IntPtr(LVSICF_NOSCROLL | LVSICF_NOINVALIDATEALL); break; case WM_REFLECT + WM_NOTIFY: var nmhdr = Marshal.PtrToStructure(m.LParam); // Ctrl+クリックで選択状態を変更した場合にイベントが発生しない問題への対処 if (nmhdr.code == LVN_ODSTATECHANGED) this.OnSelectedIndexChanged(EventArgs.Empty); break; } try { base.WndProc(ref m); } catch (ArgumentOutOfRangeException) { //Substringでlengthが0以下。アイコンサイズが影響? } catch (AccessViolationException) { //WndProcのさらに先で発生する。 } if (this.IsDisposed) return; if (vPos != -1) if (vPos != NativeMethods.GetScrollPosition(this, NativeMethods.ScrollBarDirection.SB_VERT)) VScrolled?.Invoke(this, EventArgs.Empty); if (hPos != -1) if (hPos != NativeMethods.GetScrollPosition(this, NativeMethods.ScrollBarDirection.SB_HORZ)) HScrolled?.Invoke(this, EventArgs.Empty); } } }