因為項目原因,需要實作類似于安裝程式裡面的上一步下一步,而且還要跳步,最開始實作方式是把每個步奏設定為一個獨立的窗體,然後show,close,代碼可想而知,後來靈光一閃想到用tabControl,正準備大幹一場時,發現這貨居然表頭不能隐藏,用了一些别扭的方法後勉強隐藏了,又發現這貨背景色不能設定為透明。。。好在天無絕人之路,前幾天搜到了大名鼎鼎的CSKIN(請原諒我的無知。。。),用了插件帶的tabcontrol,發現可以背景色透明,再加上這幾天搜尋的結果,于是想能不能擴充一下,達到需求,廢話少說,直接上代碼
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Collections;
using System.Windows.Forms;
namespace WindowsFormsApplication1
{
/// <summary>
/// 檔案名:TabControlUltimate
/// 描述:基于SkinTabControl的頁籤,隐藏頭部,支援子頁籤load事件
/// 作者:Bright
/// 日期:2017/4/27 22:48:33
/// 修改記錄:
/// R1:
/// 作者
/// 日期
/// 原因
/// R2:
/// 作者
/// 日期
/// 原因
/// </summary>
public class TabControlUltimate : CCWin.SkinControl.SkinTabControl
{
protected override void OnCreateControl()
{
if( !this.DesignMode && TabCount > 0 )
{
System.Windows.Forms.Panel p = new System.Windows.Forms.Panel();
p.SuspendLayout();
this.SuspendLayout();
p.Parent = this.Parent;
p.BackColor = System.Drawing.Color.Transparent;
p.Size = new System.Drawing.Size( this.Size.Width, this.TabPages[0].Size.Height );
p.Location = new System.Drawing.Point( this.Location.X, this.Location.Y + this.Size.Height - this.TabPages[0].Size.Height );
this.Parent = p;
this.Location = new System.Drawing.Point( 0, this.TabPages[0].Size.Height - this.Size.Height );
this.ResumeLayout( false );
p.ResumeLayout( false );
}
base.OnCreateControl();
}
protected override void OnPaint( System.Windows.Forms.PaintEventArgs e )
{
if( this.DesignMode )
{
}
base.OnPaint( e );
}
protected override void OnKeyDown( KeyEventArgs e )
{
// 屏蔽control+tab
if(e.Modifiers == Keys.Control&& e.KeyCode == Keys.Tab )
{
e.Handled = true;
return; //trap CTRL+TAB
}
else if( e.Modifiers == (Keys.Control | Keys.Shift) && e.KeyCode == Keys.Tab )
{
e.Handled = true;
return; //trap CTRL+SHIFT+TAB
}
else
{
base.OnKeyDown( e );
}
}
/// <summary>
/// 顯示索引值的頁籤
/// </summary>
/// <param name="i"></param>
/// <returns>null成功,非null表示失敗原因</returns>
public virtual string ShowIndex( int i )
{
if( i < 0 || i>= TabPages.Count )
{
return "out of index";
}
SelectedIndex = i;
return null;
}
/// <summary>
/// 顯示索引值的頁籤
/// </summary>
/// <param name="tab_name">子頁籤Name</param>
/// <returns>null成功,非null表示失敗原因</returns>
public virtual string ShowIndex(string tab_name)
{
if( TabPages.ContainsKey(tab_name) )
{
this.SelectTab( tab_name );
return null;
}
else
{
return "out of index";
}
}
/// <summary>
/// 顯示下一個頁籤
/// </summary>
public virtual void ShowNext()
{
ShowIndex( SelectedIndex + 1 );
}
/// <summary>
/// 顯示上一個頁籤
/// </summary>
public virtual void ShowPre()
{
ShowIndex( SelectedIndex - 1 );
}
/// <summary>
/// 初始化頁籤控件
/// </summary>
public virtual void InitTab()
{
SelectedIndex = 0;
}
// 用于記錄頁籤有沒有選中過
private Hashtable selected_time = new Hashtable();
protected override void OnSelectedIndexChanged( EventArgs e )
{
base.OnSelectedIndexChanged( e );
if( !selected_time.ContainsKey( SelectedTab.Name ) )
{
EventHandler eh = (EventHandler)event_list[SelectedTab.Name];
if( eh!= null )
{
eh( event_list[SelectedTab.Name], e );
}
}
selected_time[SelectedTab.Name] = 1;
}
private Hashtable event_list = new Hashtable();
/// <summary>
/// 為子頁籤添加第一次選中的事件,需要在初始化前使用
/// </summary>
/// <param name="tab_name">子頁籤Name</param>
/// <param name="e">事件參數</param>
public void TabPageLoadBind(string tab_name, EventHandler e)
{
event_list[tab_name] = e;
}
}
}
調用示例
private void Form1_Load( object sender, EventArgs e )
{
mytab1.TabPageLoadBind( "skinTabPage3", ( obj, ee ) =>
{
MessageBox.Show( "skinTabPage3" );
} );
mytab1.TabPageLoadBind( "skinTabPage4", ( obj, ee ) =>
{
MessageBox.Show( "skinTabPage4" );
} );
// 初始化之前進行事件綁定
mytab1.InitTab();
}
這個控件可以為子頁籤添加類似于load的事件,而且還屏蔽了用快捷鍵方式切換tab,雖然叫ultimate,但還不是最終版,最終版應該是改造tabpage,把事件加在tabpage上,然後還要為新的tabcontrol配置設定設計器任務,不過這個超出了本人的能力,現在這個已經滿足要求了,有知道最終版做法的請說明一下,謝謝