重新绘制TabControl的Tabpage标签,添加图片及关闭按钮

重新绘制TabControl的Tabpage标签,添加图片及关闭按钮
1 using System;
  2 using System.Collections.Generic;
  3 using System.ComponentModel;
  4 using System.Data;
  5 using System.Drawing;
  6 using System.Linq;
  7 using System.Text;
  8 using System.Windows.Forms;
  9 
 10 
 11 namespace TabControlTest
 12 {
 13     public partial class Form1 : Form
 14     {
 15         public Form1()
 16         {
 17             InitializeComponent();
 18         }
 19 
 20         const int CLOSE_SIZE = 15;
 21 //tabPage标签图片
 22         Bitmap image = new Bitmap("E:\\1\\2.jpg");
 23 //绘制“X”号即关闭按钮
 24         private void MainTabControl_DrawItem(object sender, DrawItemEventArgs e)
 25         {
 26 
 27             try
 28             {
 29                 Rectangle myTabRect = this.MainTabControl.GetTabRect(e.Index);
 30 
 31                 //先添加TabPage属性   
 32                 e.Graphics.DrawString(this.MainTabControl.TabPages[e.Index].Text
 33                 , this.Font, SystemBrushes.ControlText, myTabRect.X + 2, myTabRect.Y + 2);
 34 
 35                 //再画一个矩形框
 36                 using (Pen p = new Pen(Color.White))
 37                 {
 38                     myTabRect.Offset(myTabRect.Width - (CLOSE_SIZE + 3), 2);
 39                     myTabRect.Width = CLOSE_SIZE;
 40                     myTabRect.Height = CLOSE_SIZE;
 41                     e.Graphics.DrawRectangle(p, myTabRect);
 42                  
 43                 }
 44 
 45                 //填充矩形框
 46                 Color recColor = e.State == DrawItemState.Selected ? Color.White : Color.White;
 47                 using (Brush b = new SolidBrush(recColor))
 48                 {
 49                     e.Graphics.FillRectangle(b, myTabRect);
 50                 }
 51 
 52                 //画关闭符号
 53                 using (Pen objpen = new Pen(Color.Black))
 54                 {
 55                     //"\"线
 56                     Point p1 = new Point(myTabRect.X + 3, myTabRect.Y + 3);
 57                     Point p2 = new Point(myTabRect.X + myTabRect.Width - 3, myTabRect.Y + myTabRect.Height - 3);
 58                     e.Graphics.DrawLine(objpen, p1, p2);
 59 
 60                     //"/"线
 61                     Point p3 = new Point(myTabRect.X + 3, myTabRect.Y + myTabRect.Height - 3);
 62                     Point p4 = new Point(myTabRect.X + myTabRect.Width - 3, myTabRect.Y + 3);
 63                     e.Graphics.DrawLine(objpen, p3, p4);
 64                       ////=============================================
 65                     Bitmap bt = new Bitmap(image);
 66                     Point p5 = new Point(myTabRect.X-50, 4);
 67                     e.Graphics.DrawImage(bt, p5);
 68                     //e.Graphics.DrawString(this.MainTabControl.TabPages[e.Index].Text, this.Font, objpen.Brush, p5);
 69                 }
 70 
 71 
 72                 //绘制小图标              
 73                 //==============================================================================
 74                 //Bitmap bt = new Bitmap("E:\\1\\2.jpg");
 75                 //Point p5 = new Point(4, 4);
 76                 ////e.Graphics.DrawImage(bt, e.Bounds);
 77                 //e.Graphics.DrawImage(bt, p5);
 78                 //Pen pt = new Pen(Color.Red);
 79                 ////e.Graphics.DrawString(this.MainTabControl.TabPages[e.Index].Text, this.Font, pt.Brush, e.Bounds);
 80                 //e.Graphics.DrawString(this.MainTabControl.TabPages[e.Index].Text, this.Font, pt.Brush, p5);
 81 
 82                 e.Graphics.Dispose();
 83             }
 84             catch (Exception)
 85             {
 86 
 87             }
 88 
 89 
 90         }
 91         //=======================================================================
 92      
 93 //关闭按钮功能
 94         private void MainTabControl_MouseDown(object sender, MouseEventArgs e)
 95         {
 96             if (e.Button == MouseButtons.Left)
 97             {
 98                 int x = e.X, y = e.Y;
 99 
100                 //计算关闭区域   
101                 Rectangle myTabRect = this.MainTabControl.GetTabRect(this.MainTabControl.SelectedIndex);
102 
103                 myTabRect.Offset(myTabRect.Width - (CLOSE_SIZE + 3), 2);
104                 myTabRect.Width = CLOSE_SIZE;
105                 myTabRect.Height = CLOSE_SIZE;
106 
107                 //如果鼠标在区域内就关闭选项卡   
108                 bool isClose = x > myTabRect.X && x < myTabRect.Right
109                  && y > myTabRect.Y && y < myTabRect.Bottom;
110 
111                 if (isClose == true)
112                 {
113                     this.MainTabControl.TabPages.Remove(this.MainTabControl.SelectedTab);
114                 }
115             }
116 
117 
118         }
119 //初始化页面
120         private void Form1_Load(object sender, EventArgs e)
121         {
122             //清空控件
123             //this.MainTabControl.TabPages.Clear();
124             //绘制的方式OwnerDrawFixed表示由窗体绘制大小也一样
125             this.MainTabControl.DrawMode = TabDrawMode.OwnerDrawFixed;
126             this.MainTabControl.Padding = new System.Drawing.Point(CLOSE_SIZE, CLOSE_SIZE);
127             this.MainTabControl.DrawItem += new DrawItemEventHandler(this.MainTabControl_DrawItem);
128             this.MainTabControl.MouseDown += new System.Windows.Forms.MouseEventHandler(this.MainTabControl_MouseDown);
129         }
130 
131  //添加新的tabPage并修改所有tabPage标签上面的图片
132         private void button1_Click(object sender, EventArgs e)
133         {
134             TabPage tabtage=new TabPage ();
135             MainTabControl.TabPages.Add(tabtage);
136             MainTabControl.SelectedTab = tabtage;
137             image = new Bitmap("E:\\1\\3.jpg");
138 
139 
140         }
141 //关闭选中的tabPage
142         private void button2_Click(object sender, EventArgs e)
143         {
144             MainTabControl.TabPages.Remove(MainTabControl.SelectedTab);
145         }
146 
147      
148 
149     }
150 }
View Code

制作个性Tabcontrol控件(收藏~~)

重新绘制TabControl的Tabpage标签,添加图片及关闭按钮
有的人认为,Blend主要是用来修改一下颜色,调整一下布局之类的,大部分的SL开发还是用VS搞定;这样的结果我们可以预测,他的UI一定是很普通的,也许布局颜色搭配之类算不上难看,符合一般审美,但是绝不是高水平的UI,也不会是销售人员满意的设计;
 有的人认为,Blend太难学了,总是搞不懂应该怎么去操作,感觉跟其他设计工具都有点不一样,所以,很多人对它很感兴趣,希望用它来实现自己心中如诗如画的构思;于是站在门口,不断的徘徊,却始终不得其门;
 我其实不是UI设计人员,我连Photoshop都不会,也从没有设计过像样的东西;并且对于Silverlight,我也不是很有经验;我想告诉你的是,它没有那么复杂,因为像我这样的人都能捣弄一下,除非你在后面留言说:“楼主,我比你想象的要笨! ” 
示例说明:
 作为Blend操作的入门篇,我对Blend操作几个阶段的的定义是:
 1. 入门阶段-能得心应手的修改任何控件的样式;
 2. 中级阶段-能得心应手的设计动画,设计自定义控件的模板,状态,状态过渡,各种Behavior的运用和开发
 3. 高级阶段-只要你想得到的场景,你都知道怎么去设计
 这篇是针对第一阶段的朋友写的,因为我也只是刚过了第一阶段,其实开始不知道要做什么样的示例比较合适。昨天群里有一MM问怎么做一个好看一点的Tabcontrol控件,于是就想以修改Tabcontrol模板为示例,顺便也可以给她的帮助。以下是示例截图,先看一下效果,我不是设计人员,不要对我要求太高,我就做这个简单的修改,你会比我设计的好看得多:
 
小话设计模式:
 有一个坏毛病,总是喜欢在文章中插一些不相关的东西,这里也来一个小话设计模式。
 记得,以前写Behavior的时候也是提到了设计模式,那么这里又提到设计模式。有的人说,既然对设计模式这么感兴趣,这个人为什么没有写过一篇设计模式的文章呢,也来讲一下比如单例模式的UML图是什么样子,实现代码是什么样子。
 等等,这时候你会说,那些设计模式书上不是已经这样写了么!
 是的,个人觉得,对于那些经典设计模式的描述,已经有很多书了,而且我认为没有必要再出那些设计模式的书了,除非你自己总结出新的设计模式实践,或者真正实用的应用。大部分的关于设计模式的书,都是去构造一个不同的故事,不同的场景,最后写出相同的标准的类,接口的定义,一个简单的差不多都相同代码的示例。
 我建议再也不要写这样的书了,很多人看了很多本这样的书,还是不会应用(我就是其中一个)。
 我建议的下一本关于设计模式的书是,用真实的经典的开发者比较熟悉的案例来分析设计模式。那么我们来看一下模板方法。
 不知道你现在对模板方法有一个什么样的概念,如果你现在还不是很清楚,那么我相信这一次你应该可以清楚了,不清楚的请在后面留言“我很笨!”
 可能在你的开发中很少用到模板方法,但是我告诉你你天天在运用模板方法,你相不相信呢?
 模板方法的概念是,定义一个框架,但是将具体的算法延迟到子类中,使得子类可以定义自己的算法实现。这样的定义像很多设计模式书描述的一样,你会很迷糊,下面举个例子:
 ASP.NET的核心机制就是模板方法,每一个Page是一个实现IHttpHandler的实现类,Page定义了一些框架,比如page_Load这些方法,这样当ASP.NET在执行页面请求的时候,首先找到对应的Page(也就是子类),然后依次调用框架中定义的方法,比如Init,Pre_Render等等。ASP.NET框架本身来控制对子类方法的调用,通过这样的方式来处理不同的请求,你可以定义不同的Page做不同的事情,用不同的算法,后台逻辑。
 所以,模板方法是非常重要的设计模式,它通常用于框架设计,你定义一个框架,不同的子类遵循这个框架实现规定的方法,然后框架主体来调用子类的方法,到达可扩展性,因为子类只要遵循这个框架定义,它可以有不同的实现。
 所以,模板方法不仅仅只是一种设计模式,它基本上可以算作一个架构风格了,你可以站在架构师的层面,去运用模板方法。
 那我们这里为什么讲模板方法呢?因为Silverlight实现控件模板自定义的样式的机制也是模板方法:控件定义一个ControlTemplate属性,这个对象你可以自己定义,如果你不定义,Silverlight去取默认的Xaml样式,实例化然后赋值给控件,如果你定义了自己的ControlTemplate对象,Silverlight就用你自定义的对象来初始化控件,这样控件就能使用你的样式。
 Silverlight中的ControlTemplate就相当于Page,其本身相当于定义了一个框架,具体的样式定义在ControlTemplate中。控件本身不知道样式,就像ASP.NET不知道怎么去处理每个页面。
 这,就是模板方法,有了这个提醒之后,相信你会发现更多模板方法的样子,相信现在你应该在对模板方法的概念感到模糊了。
 我相信,用我们熟悉的东西来讲设计模式,这样效果好得多,这样的例子很多很多,因为经典设计模式其实也应用在很多经典的软件设计中,而不仅仅只是一个概念。
 很希望,下一本设计模式的书是这样的。
Blend入门很简单:
 除了修改颜色等基本属性,下面列举几点必须掌握的东西:
 1. 示例数据
 示例数据的管理位于Data面板,如下图:
 
 为什么它是很重要的,设计Silverlight UI很痛苦的是,数据都是从Service上获取的,这样在设计的时候一般看不到控件的外观。那么这个时候我们就要采用本地示例数据,它能够帮助我们在设计时很完整的看到运行时的效果,这样才能更好的设计。它的原理是,Blend或者VS的设计器在呈现Xaml的时候,其实和真实的运行时差不多,在运行时,Silverlight插件初始化SL控件,获取每个元素的布局信息,然后将其绘制到屏幕上正确的位置。那设计器想要呈现设计时的外观,也是要家里这样的过程,想要初始化控件树,绘制到屏幕,不同的是设计时只会初始化Xaml中定义的元素。如果这个时候示例数据是本地资源,并且示例数据的集合对象是放在Xaml中初始化的,那么设计器就能找到对应数据,将它作为控件的Context,而到该控件实例化的时候,它发现已经有了数据,自然不会放过,会将其绘制出来,所以我们在设计时能看到效果。尤其是某些复杂的集合控件,这非常重要。Blend中示例数据实际上是一个Xml文件,可以定义不同类型的数据,比如Image,String,你告诉它想要多少条数据,它就会生成随机的示例数据,达到以假乱真:
 示例数据的使用非常简单,拖动collection那一行,至一个控件比如ListBox即可:
 然后就会生成预览图:
 这个时候再来修改模板是不是方便多了,一般建议是建立和真实的数据结构一样的属性,在真实运行的时候,将数据源改为真实的数据源,这样绑定属性不用改。这是Blend入门最基本的东西。
 
 2. 怎样调出控件模板ControlTemplate
 在Silverlight中修改模板得首先把原来的模板样式取出来,然后在其基础之上去修改,Blend为我们提供了一种很方便的方式把这段模板取出来,如我们想调出ListBox的ControlTemplate,如下:
 
 右键选择Edit Template->Edit a Copy,Blend就会把控件的默认模板xaml定义片段复制到Xaml中,你可以在此基础之上修改,你也可以选中ListBox,在视图面板中进行该操作,效果是一样的,建议采用后一种方式,后面会看到为什么:
 这个时候你就可以看到该模板的定义,状态,可以进行修改。这里有一个细节就是,怎样退出控件模板,即怎样在页面也控件模板之间切换,如下我们看到的是ListBox的ControlTemplate视图,我们可以点击视图面板上端的控件名称切换为页面视图。Blend所有的模板之间的切换都是在这里,如果想进入模板视图,可以再选择:ListBox->Edit Template->Edit Current进入:
 
 3. 怎样调出集合控件中子元素的ControlTemplate,即集合控件的ItemTemplate
 按如上的操作就可以进入ItemTemplate视图,这个时候就可以编辑每个子项的模板。不过这里我们将看到,这里的只包含每个子项的组成元素,每个子项包含哪些元素。但是不能编辑不同的状态,比如ListBox在MouseOver的时候是什么样式,在Selected的时候是另外一种样式,这个时候需要选择下面的Edit Generated Item Container(ItemContainerStyle),就可以编辑控件的状态。
 
 4. 怎样编辑控件的不同状态下的外观
 我们看到,经典的Button控件,它不是只有一个样子的。它有很多状态,比如鼠标经过是一种样子,鼠标离开,点击都会呈现不同的样子,这都是通过定义不同的状态来实现的。
 一个控件可以定义多个状态,在每个状态下可以设定不同的外观,比如ComboBox控件,在点击下拉框的时候就显示下面的选项,鼠标离开就隐藏选项。当进入不同的状态,状态管理器就将改变控件外观至该状态定义的样子。关于状态可以参考我之前的文章。我们看下ListBoxItem的状态:
 状态位于States面板,我们看到ListBoxItem模板定义了很多状态,状态可以分成不同的状态组,每个状态组之间的状态是互斥的,选择其中一个状态,就可以设置该状态下控件的外观,比如如图我点击最后一个Selected状态,这个时候视图面板就会多出一个红色的框,表示开始录制动画,在运行时的如果切换至该状态,状态管理器就会触发动画使外观呈现定义的样子。比如我们将每个项的宽度拉长,这样在运行时选择的时候,被选中的项就会拉长。
 
示例操作:
 上面介绍的几点是我觉得Blend入门应该知道的最基本的东西,当然不能写的太详细,需要你去操作体会实践,我也不可能写出所有细节,每一个操作都还有一些更细的东西和功能,可以自己去摸索,我们下面就来从头实现本文提到的示例:
 1. 打开Blend 4,新建MyTabcontrol项目,项目类型Silverlight Application + WebSite
 2. 添加一个TabControl到MainPage控件中,并设置MainPage背景为黑色
 3. 选中TabControl控件,在Properties面板设置如下,并这个时候TabControl呈现如图所示外观:
 4. 右键选择TabControl—》Edit Template—》Edit a Copy,这个时候会看到如下图,删掉其中的后面三项,只留下TemplateTop(后面三项是说Tab分别在下面,左边和右边的情形,我们这里Tabs在上边,其他三个用不着,你也可以选择其他的布局):
 
 5. 选择TemplateTop,可以看到其是一个Grid控件,共有2行,第一行是Tabs,第二行就是下面的现实面板。修改Properties面板中RowDefinitions ,将Grid的第一行高度为50px;
 6. 展开TemplateTop,选中TabPanelTop,这就是定义TabControl控件上面Tabs部分的背景的。在Properties面板中设置Background为:#FFBBD7FA 
 7. 选中下面的Border,将Background Reset: #FFB1CBEB;BorderBrush:#FFE4E4E4;BorderThickness:0,0,1,0;并加上圆角CornerRadius:0,0,10,10;
 8,. 选中ContentTop,加上Margin:10。这个时候看起来是这个样子:
 9. 下面我们来编辑TabItem的模板。
 回到MainPage的视图,展开TabControl,选择第一个TabItem,调出其面板(右键选择TabItem—>Edit Template—>Edit a Copy)。进入TabItem面板之后看到有许多部件,删掉其他的只留下三个:TemplateTopSelected,TemplTopUnSelected,FocusVisualElement,原因和前面的一样,我们的Tab只会在顶端,你可以自己去修改其他样式。
 接下来的,TemplateTopSelected就是定义选择状态的外观,TemplateTopUnSelected就是定义没有选择的外观。
 10. 在States面板,选中base,在Objects and TimeLine面板选中Root元素,在属性面板修改高度为50px;
 11. 选中Selected状态,开始录制
 12. 展开TemplateTopSelected,将b元素order的Margin,BorderThickness,CornerRadius都设置为0,Background设置为:#FFBBD7FA
 13. 选择border1元素,设置参数,BorderThickness:1,1,1,0;CornerRadius:10,10,10,0;Margin:2,2,5,0;Background和BorderBrush都设置为纯白色。
 14. 选择HeaderTopSelected元素:修改字体大小样式至以下的样子:这个时候看起来是这样:
 15. 接下来我们修改TemplateTopUnSelected的样子,它是处于UnSelected的状态,所以我们在States面板选中UnSelected状态,开始录制。注意这个时候找到TabControl控件中的两个TabItem控件,将第一个TabItem的Style="{StaticResource TabItemStyle1}"样式复制给第二个。
 16. 对着前面的步骤,修改TemplateTopUnSelected下面的各个元素,使之呈现如下样式:
 17. 大功告成,就是还有一点小小的瑕疵,就是在选中的时候有个讨厌的边框,这个时候将FocusVisualTop的BorderBrush设置为无色就可以了,看一下我们的成果:
 
总结:
 本篇先总结了一点Blend入门的注意事项和要领,接着展示了一个示例。
 开始使用Blend,绝大部分是修改控件模板,本文抛砖引玉,演示了一般的修改控件模板的方法,包括ControlTemplate以及集合控件的ItemTemplate。掌握了这些方法,修改Silverlight控件外观就不难了,相信,杀掉Silverlight默认的丑陋的外观,你自己设计的外观一定如诗如画。你们公司的销售也该请你吃饭了,哈哈。
 当然,Blend还有很多东西,本文讲得很简单,还希望大家多多实践,多看看老外的文章,示例。
 后期有时间在写一篇高级一点的,包括动画和自定义状态,以及状态之间的转换动画等等。
 总之,Blend还是比较有趣的,只要你想得出来的,它基本上都可以做出来,希望大家多多实践。
 如有兴趣,可以加入QQ群讨论,本群热烈欢迎有志于高水平UI设计的朋友,当然我们也讨论Silverlight,Windows Azure,Ajax,WCF等等微软技术,欢迎你。
 QQ群:6183299
View Code

C# TabContorl选项卡

重新绘制TabControl的Tabpage标签,添加图片及关闭按钮
C# TabContorl选项卡
if(textBox1.Text=="")
{
tabControl1.SelectedIndex=0;//如果想打开的是第二个,则为1
}
这样就可以打开你想打开的任意一个对话框
想随着选择combobox地具体内容地不同,TabControl顶出现地标签也不同,标签已经创建好地,如1共就为三个标签页,需要么就为三个全显示,需要么就为显示其中地两个,应该如何?
我还没有找到好的办法,我想只想将其中不用的移除吧。
tabControl1.TabPages.Remove(tbOneCinema);其中tbOneCinema为当中一个选项卡的(Name)
添加新的项卡时,为
TabPage MyTabPage=new TabPages(Title);
tabControl1.TabPages.Add(MyTabPage);
c#中字符串截取使用的方法
String.Substring 方法
名称 说明 
String.Substring (Int32)         从此实例检索子字符串。子字符串从指定的字符位置开始。 
String.Substring (Int32, Int32) 从此实例检索子字符串。子字符串从指定的字符位置开始且具有指定的长度。
举例如下:
using System;
using System.Collections.Generic;
using System.Text;
namespace ConsoleApplication1
{
     class Program
     {
         static void Main(string[] args)
         {
             string s = "Hello C# World!";
             //s1为从s中截取的位置为3的字符以后的字符子串,3表示子字符串的起始字符位置
            string s1=s.Substring(3);
             Console.WriteLine(s1);
             //s2为从s中截取的位置为6的字符开始长度为2的字符串,6表示子字符的起始字符位置,2表示子字符长度
            string s2 = s.Substring(6, 2);
             Console.WriteLine(s2);
        }
     }
}
结果如下:
lo C# World!
C#
随即在附上一个C#截取字符串函数
public string getString(string RawString, Int32 Length)
     {
         if (RawString.Length <= Length)
         {
             return RawString;
         }
         else
         {
             for (Int32 i = RawString.Length - 1; i >= 0; i--)
             {
                 if (System.Text.Encoding.GetEncoding("GB2312").GetByteCount(RawString.Substring(0, i)) < Length)
                 {
                     return RawString.Substring(0, i) + "...";
                 }
             }
             return "...";
         }
     }

1/**//// <summary> 
2         /// 截取字符串,不限制字符串长度 
3         /// </summary> 
4         /// <param name="str">待截取的字符串</param> 
5         /// <param name="len">每行的长度,多于这个长度自动换行</param> 
6         /// <returns></returns> 
7         public string CutStr(string str,int len) 
8         {     string s=""; 
9             
10             for(int i=0;i<str.Length ;i++) 
11             { 
12                 int r= i% len; 
13                 int last =(str.Length/len)*len; 
14                 if (i!=0 && i<=last) 
15                 { 
16                     
17                     if( r==0) 
18                     { 
19                         s+=str.Substring(i-len,len)+"<br>"; 
20                     } 
21                         
22                 } 
23                 else if (i>last) 
24                 { 
25                     s+=str.Substring(i-1) ; 
26                     break; 
27                 } 
28                 
29             } 
30             
31             return s; 
32             
33         } 
34 
35 
36         /**//// <summary> 
37         /// 截取字符串并限制字符串长度,多于给定的长度+。。。 
38         /// </summary> 
39         /// <param name="str">待截取的字符串</param> 
40         /// <param name="len">每行的长度,多于这个长度自动换行</param> 
41         /// <param name="max">输出字符串最大的长度</param> 
42         /// <returns></returns> 
43         public string CutStr(string str,int len,int max) 
44         { 
45             string s=""; 
46             string sheng=""; 
47             if (str.Length >max) 
48             { 
49                 str=str.Substring(0,max) ; 
50                 sheng=""; 
51             } 
52             for(int i=0;i<str.Length ;i++) 
53             { 
54                 int r= i% len; 
55                 int last =(str.Length/len)*len; 
56                 if (i!=0 && i<=last) 
57                 { 
58                     
59                     if( r==0) 
60                     { 
61                         s+=str.Substring(i-len,len)+"<br>"; 
62                     } 
63                         
64                 } 
65                 else if (i>last) 
66                 { 
67                     s+=str.Substring(i-1) ; 
68                     break; 
69                 } 
70                 
71             } 
72             
73             return s+sheng; 
74             
75         } 
 
View Code

 

上一篇:P177-解决吸顶效果-TabControl的offsetTop获取分析


下一篇:【C#/WinForm】Tabcontrol中添加TabPage,并对TabPage进行控制