[原創][開源] SunnyUI.Net 開發日誌:ListBox 增加跟隨滑鼠滑過高亮

来源:https://www.cnblogs.com/yhuse/archive/2020/05/21/12933885.html
-Advertisement-
Play Games

SunnyUI.Net 開發日誌:ListBox 增加跟隨滑鼠滑過高亮,這可是在別人兩年的基礎上再改進的,哈哈!~ ...


 

SunnyUI.Net, 基於 C# .Net WinForm 開源控制項庫、工具類庫、擴展類庫、多頁面開發框架

 

 SunnyUI.Net 開發日誌:ListBox 增加跟隨滑鼠滑過高亮

 

 QQ群里,寸木說,ListBox滑鼠移動時,當前行需要焦點,我想了想,不難實現啊

不就是在滑鼠移動時重繪Item嘛,何況選中的Item已經改了顏色了。

見UIListBox代碼:

protected override void OnDrawItem(DrawItemEventArgs e)
        {
            base.OnDrawItem(e);
            BeforeDrawItem?.Invoke(this, Items, e);
            if (Items.Count == 0)
            {
                return;
            }

            e.DrawBackground();

            if (e.Index < 0 || e.Index >= Items.Count)
            {
                return;
            }

            StringFormat sStringFormat = new StringFormat();
            sStringFormat.LineAlignment = StringAlignment.Center;

            Color backColor = (e.State & DrawItemState.Selected) == DrawItemState.Selected ? ItemSelectBackColor : BackColor;
            Color foreColor = (e.State & DrawItemState.Selected) == DrawItemState.Selected ? ItemSelectForeColor : ForeColor;

            Rectangle rect = new Rectangle(e.Bounds.X, e.Bounds.Y, e.Bounds.Width - 1, e.Bounds.Height - 1);
            e.Graphics.FillRectangle(BackColor, e.Bounds);
            e.Graphics.FillRoundRectangle(backColor, rect, 5);
            e.Graphics.DrawString(Items[e.Index].ToString(), e.Font, foreColor, e.Bounds, sStringFormat);
        }

看:(e.State & DrawItemState.Selected) == DrawItemState.Selected 選中行狀態嘛

看了e.State有e.State == DrawItemState.HotLight,不就是高亮的麽,於是開始擼代碼,加狀態判斷

Run

暈。。。沒變,這HotLight不起作用,好吧,問度娘。。。

翻山越嶺,跋山涉水。。。

找到這篇:https://www.jb51.cc/csharp/101121.html

其中提到:

   我在我的WinForms應用程式中使用OwnerDrawFixed作為DrawMode用於自定義ListBox控制項.當用戶將滑鼠懸停在列表框項目上時,我希望重新繪製ListBoxItem的背景(或執行其他操作),即在MouseMove …DrawItemState.HotLight永遠不適用於ListBox,所以我想知道如何模擬它,如何解決這個問題.

DrawItemState.HotLight永遠不適用於ListBox,是永遠。。。怎麼這麼遠。。。

繼續往下看:

解決方法

我花了兩年時間為你找到答案,但這裡是:

DrawItemState.HotLight僅適用於所有者繪製的菜單,而不適用於列表框.對於ListBox,您必須自己跟蹤項目:

public partial class Form1 : Form
{
  private int _MouseIndex = -1;
 
  public Form1()
  { InitializeComponent(); }
 
  private void listBox1_DrawItem(object sender,DrawItemEventArgs e)
  {
    Brush textBrush = SystemBrushes.WindowText;
 
    if (e.Index > -1)
    {
      if (e.Index == _MouseIndex)
      {
        e.Graphics.FillRectangle(SystemBrushes.HotTrack,e.Bounds);
        textBrush = SystemBrushes.HighlightText;
      }
      else
      {
        if ((e.State & DrawItemState.Selected) == DrawItemState.Selected)
        {
          e.Graphics.FillRectangle(SystemBrushes.Highlight,e.Bounds);
          textBrush = SystemBrushes.HighlightText;
        }
        else
          e.Graphics.FillRectangle(SystemBrushes.Window,e.Bounds);
      }
      e.Graphics.DrawString(listBox1.Items[e.Index].ToString(),e.Font,textBrush,e.Bounds.Left + 2,e.Bounds.Top);
    }
  }
 
  private void listBox1_MouseMove(object sender,MouseEventArgs e)
  {
    int index = listBox1.IndexFromPoint(e.Location);
    if (index != _MouseIndex)
    {
      _MouseIndex = index;
      listBox1.Invalidate();
    }
  }
 
  private void listBox1_MouseLeave(object sender,EventArgs e)
  {
    if (_MouseIndex > -1)
    {
      _MouseIndex = -1;
      listBox1.Invalidate();
    }
  }
}

兄弟們,人家這花兩年時間解決的,應該有用,繼續再找找,又找到一篇洋文的:

https://stackoverflow.com/questions/1316027/listbox-drawitem-hotlight-state-in-the-ownerdraw-mode

It took me only two years to find the answer for you, but here it is:

The DrawItemState.HotLight only applies to owner drawn menus, not the listbox.

For the ListBox, you have to keep track of the item yourself:

看看,也是兩年,估計上面中文的從這個翻譯過來。

除了這倆,還真沒找到。

繼續擼代碼,果真管用。不過還是有問題,滑鼠滑快了,ListBox閃爍的厲害。

分析代碼  listBox1.Invalidate(); 這是刷新全部的。滑鼠滑過也就和本次選中和上次選中的有關係。

就刷這兩個Item就行,有了思路,擼代碼三連發:

        private int lastIndex = -1;
        private int mouseIndex = -1;

        [Browsable(false)]
        public int MouseIndex
        {
            get => mouseIndex;
            set
            {
                if (mouseIndex != value)
                {
                    if (lastIndex >= 0 && lastIndex != SelectedIndex)
                    {
                        OnDrawItem(new DrawItemEventArgs(this.CreateGraphics(), Font, GetItemRectangle(lastIndex), lastIndex, DrawItemState.Grayed));
                    }

                    mouseIndex = value;
                    if (mouseIndex >= 0 && mouseIndex != SelectedIndex)
                    {
                        OnDrawItem(new DrawItemEventArgs(this.CreateGraphics(), Font, GetItemRectangle(value), value, DrawItemState.HotLight));
                    }

                    lastIndex = mouseIndex;
                }
            }
        }

        protected override void OnMouseMove(MouseEventArgs e)
        {
            base.OnMouseMove(e);
            MouseIndex = IndexFromPoint(e.Location);
        }

        protected override void OnMouseLeave(EventArgs e)
        {
            base.OnMouseLeave(e);
            MouseIndex = -1;
        }

其中 new DrawItemEventArgs(this.CreateGraphics(), Font, GetItemRectangle(value), value, DrawItemState.HotLight) 

第一個參數Graphics,也找了一會兒,後來看這篇:

https://www.cnblogs.com/yuanyeguhong/archive/2013/09/20/3330606.html

 

其實就是Graphics對象的DrawString方法,而參數e中的Graphics是如何來的呢。
我們接著分析DrawItemEventArgs這個類,他既然是對listBox1某一項的屬性的打包,
那麼我估計其中的Graphics對象就是由listBox1.creatgraphics而來的。
好了,到此我們就可以自定義重繪listbox某項的函數了,可任意調用的哦!

好了,至此,問題都已找到答案,再捋一下思路,把邏輯理順。擼代碼,調試,OK!!!

主要代碼如下:

protected override void OnDrawItem(DrawItemEventArgs e)
        {
            base.OnDrawItem(e);

            BeforeDrawItem?.Invoke(this, Items, e);
            if (Items.Count == 0)
            {
                return;
            }

            bool otherState = e.State == DrawItemState.Grayed || e.State == DrawItemState.HotLight;
            if (!otherState)
            {
                e.DrawBackground();
            }

            if (e.Index < 0 || e.Index >= Items.Count)
            {
                return;
            }

            StringFormat sStringFormat = new StringFormat();
            sStringFormat.LineAlignment = StringAlignment.Center;

            bool isSelected = (e.State & DrawItemState.Selected) == DrawItemState.Selected;
            Color backColor = isSelected ? ItemSelectBackColor : BackColor;
            Color foreColor = isSelected ? ItemSelectForeColor : ForeColor;

            Rectangle rect = new Rectangle(e.Bounds.X, e.Bounds.Y, e.Bounds.Width - 1, e.Bounds.Height - 1);
            if (!otherState)
            {
                e.Graphics.FillRectangle(BackColor, e.Bounds);
                e.Graphics.FillRoundRectangle(backColor, rect, 5);
                e.Graphics.DrawString(Items[e.Index].ToString(), e.Font, foreColor, e.Bounds, sStringFormat);
            }
            else
            {
                if (e.State == DrawItemState.Grayed)
                {
                    backColor = BackColor;
                    foreColor = ForeColor;
                }

                if (e.State == DrawItemState.HotLight)
                {
                    backColor = HoverColor;
                    foreColor = ForeColor;
                }

                e.Graphics.FillRectangle(BackColor, e.Bounds);
                e.Graphics.FillRoundRectangle(backColor, rect, 5);
                e.Graphics.DrawString(Items[e.Index].ToString(), e.Font, foreColor, e.Bounds, sStringFormat);
            }

            AfterDrawItem?.Invoke(this, Items, e);
        }

看,DrawItemState.HotLight咱也給實現了,DrawItemState.Grayed 我是隨便選的狀態,區別於其他。

想看全部代碼,看我的開源項目吧,https://gitee.com/yhuse/SunnyUI ,哎,客官別走嘛,點個Star先。

 

原創文章,轉載請保留鏈接 Sunny's blog


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 準備鏡像 我使用的centos系統,docker-ce最新版本。 先準備好三個鏡像 docker pull mysql:5.7 docker pull php:5.6-fpm docker pull nginx:latest 註意,mysql和php不要使用最新版本,坑很多。 docker imag ...
  • 在上一篇文章中我們介紹了JDK1.8的新特性有以下幾項。 1.Lambda表達式 2.方法引用 3.函數式介面 4.預設方法 5.Stream 6.Optional類 7.Nashorm javascript引擎 8.新的日期時間API 9.Base64 並且學習了JDK1.8最重要的特性--Lam ...
  • 前言 本文的文字及圖片來源於網路,僅供學習、交流使用,不具有任何商業用途,版權歸原作者所有,如有問題請及時聯繫我們以作處理。 Step1:基礎數據準備(通過爬蟲獲取到),以下是從第一期03年雙色球開獎號到今天的所有數據整理,截止目前一共2549期,balls.txt 文件內容如下 : 備註:想要現成 ...
  • 給大家分享這幾個月整理出來的Java程式員面試筆試真題,順便給大家分享一本電子書《Java程式員面試筆試寶典》需要的可以掃描文末的二維碼即可,話不多說給大家看看部分截圖。 ...
  • 這份資源我自己歷經三年才整理歸類出來,現在免費分享給大家; 面試題有:螞蟻金服、拼多多、阿裡雲、百度、唯品會、攜程、豐巢科技、樂信、軟通動力、OPPO、銀盛支付、中國平安等初,中級,高級Java面試題集合。 面試題以及答案,已經整理成PDF電子書形式打包在網盤; 面試題領取微信掃一掃,加好友請備註“ ...
  • 在秒殺,搶購等併發場景下,可能會出現超賣的現象,在 PHP 語言中並沒有原生提供併發的解決方案,因此就需要藉助其他方式來實現併發控制。 列出常見的解決方案有: 使用隊列,額外起一個進程處理隊列,併發請求都放到隊列中,由額外進程串列處理,併發問題就不存在了,但是要額外進程支持以及處理延遲嚴重,本文不先 ...
  • 6.32(游戲:贏取雙骰子賭博游戲的機會)修改編程練習題6.30使該程式運行10000次,然後顯示贏得游戲的次數 6.32(Game: chance of winning at craps)Revise Exercise 6.30 to run it 15,000 times and display ...
  • 異常有可能導致某些資源沒有釋放到,這時可以使用using或者finally來釋放資源。 有些異常的可能會間歇性的發生,不影響業務運行的,可以視為正常的邏輯,那在異常可能發生前,應該使用判斷邏輯。 if (conn.State != ConnectionState.Closed) { conn.Clo ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...