.Net Micro Framework研究—实现SideShow窗体界面

基于MF系统的Windows SideShow界面是非常炫的(如下图)。既然微软能用.Net Micro Framework实现这么棒的界面效果,我想我们也能做到。

(SideShow模拟器界面和游戏程序中的右键菜单—注意菜单弹出后,其它的界面变暗了)
现在的任务是设计一套支持鼠标(或触摸屏)的窗体框架(目前MF提供的Window类仅支持按键功能),所以正好把SideShow如此炫的界面元素也可以添加进来。
用过MF的人知道是用下面的方法来实现按键事件接收的,既然我们要支持鼠标功能,所以最好也用类似的机理实现。
   

 
  1. //按键事件  
  2.     protected override void OnButtonDown(ButtonEventArgs e)  
  3.     {  
  4.         switch (e.Button)  
  5.         {  
  6.             //按下确定键  
  7.             case Button.Select:  
  8.                 break;  
  9.             //按下左键  
  10.             case Button.Left:  
  11.                 break;  
  12.             //按下右键  
  13.             case Button.Right:  
  14.                 break;  
  15.             //按向上  
  16.             case Button.Up:  
  17.                 break;  
  18.             //按向下  
  19.             case Button.Down:  
  20.                 break;  
  21.             //按下菜单  
  22.             case Button.Menu:  
  23.                 break;  
  24.             //按下返回键  
  25.             case Button.Back:  
  26.                 break;  
  27.         }  
  28.         base.OnButtonDown(e);  
  29.     }  

用反编译工具仔细研究了MF底层库代码(.Net FrameWork 太庞大了,一个人绝对短时间内看不完,其实也很难看下去,但是对刚刚起步的MF来说,.Net Micro FrameWork就简单多了),终于理清了头绪。主要原理是在鼠标信息处理线程中通过Application.Current.Windows 属性(该属性存放了当前实例所有派生于Window类的窗体)和应用实例的this.Dispatcher属性的BeginInvoke方法,外部调用窗体鼠标事件函数。充分利用基类虚函数的妙处来实现类似按键信息处理的功能。
在YFWindowBase类中声明如下虚拟鼠标事件函数。
 

 
  1. //鼠标移动  
  2.     public virtual void OnMouseMove(object sender, MouseEventArgs e)  
  3.     {  
  4.         if (MouseMove != null) MouseMove(sender, e);  
  5.     }  
  6.     //鼠标单击  
  7.     public virtual void OnMouseClick(object sender, MouseEventArgs e)  
  8.     {  
  9.         if (MouseClick != null) MouseClick(sender, e);  
  10.     }  
  11.     //按下  
  12.     public virtual void OnMouseDown(object sender, MouseEventArgs e)  
  13.     {  
  14.        if (MouseDown != null) MouseDown(sender, e);  
  15.     }  
  16.     //抬起  
  17.     public virtual void OnMouseUp(object sender, MouseEventArgs e)  
  18.     {  
  19.         if (MouseUp != null) MouseUp(sender, e);  
  20.     }  
  21.    

在鼠标信息处理函数中执行如下的代码即可。
   

 
  1. //处理鼠标消息  
  2.     private static void TransactMouse(MouseState state, int x, int y, MouseButtons button)  
  3.     {  
  4.         if (Application.Current == nullreturn;  
  5.         for (int i = Application.Current.Windows.Count - 1; i >= 0; i--)  
  6.         {  
  7.             try 
  8.             {  
  9.                 YFWindowBase mw = Application.Current.Windows[i] as YFWindowBase;  
  10.    
  11.                 if (mw.Enabled && mw.IsVisible)  
  12.                 {  
  13.                     //判断子窗体  
  14.                     bool bReturn = false;  
  15.                     for (int j = mw.Children.Count - 1; j >= 0; j--)  
  16.                     {  
  17.                         //仅最上层并且可视的控件接收鼠标消息  
  18.                         YFControl cl = mw.Children[j];  
  19.                         if (!bReturn && cl.Visible && IsRectContains(x, y, mw.Left + cl.Left, mw.Top + cl.Top, cl.Width, cl.Height))  
  20.                         {  
  21.                             if (cl.Enable) //Enable和Visible不一样,Enable即使无效,下层控件也没有机会获得鼠标消息  
  22.                             {  
  23.                                 if (!cl._EnterFlag)  
  24.                                 {  
  25.                                     cl._EnterFlag = true;  
  26.                                     _dispatcher.BeginInvoke(new MouseInputEventHandler(cl.OnMouseEnter), cl, new MouseEventArgs(button, x - cl.Left - mw.Left, y - cl.Top - mw.Top));  
  27.                                 }  
  28.                                 if ((state & MouseState.Move) > 0)  
  29.                                     _dispatcher.BeginInvoke(new MouseInputEventHandler(cl.OnMouseMove), cl, new MouseEventArgs(button, x - cl.Left - mw.Left, y - cl.Top - mw.Top));  
  30.                                 if ((state & MouseState.Down) > 0)  
  31.                                     _dispatcher.BeginInvoke(new MouseInputEventHandler(cl.OnMouseDown), cl, new MouseEventArgs(button, x - cl.Left - mw.Left, y - cl.Top - mw.Top));  
  32.                                 if ((state & MouseState.Up) > 0)  
  33.                                     _dispatcher.BeginInvoke(new MouseInputEventHandler(cl.OnMouseUp), cl, new MouseEventArgs(button, x - cl.Left - mw.Left, y - cl.Top - mw.Top));  
  34.                                 if ((state & MouseState.Click) > 0)  
  35.                                     _dispatcher.BeginInvoke(new MouseInputEventHandler(cl.OnMouseClick), cl, new MouseEventArgs(button, x - cl.Left - mw.Left, y - cl.Top - mw.Top));  
  36.                             }  
  37.                             //向主窗体传OnMouseEvent消息,为了绘制鼠标  
  38.                             if ((state & MouseState.Event) > 0)  
  39.                                 _dispatcher.BeginInvoke(new MouseInputEventHandler(mw.OnMouseEvent), mw, new MouseEventArgs(button, x - mw.Left, y - mw.Top));  
  40.                             bReturn = true;  
  41.                         }  
  42.                         else 
  43.                         {  
  44.                             if (cl._EnterFlag)  
  45.                             {  
  46.                                 cl._EnterFlag = false;  
  47.                                 _dispatcher.BeginInvoke(new MouseInputEventHandler(cl.OnMouseLeave), cl, new MouseEventArgs(button, x - cl.Left, y - cl.Top));  
  48.                             }  
  49.                         }  
  50.                     }  
  51.                     if (bReturn) return;  
  52.                 }  
  53.    
  54.                 //仅最上层并且可视的窗体接收鼠标消息  
  55.                 if (mw.IsVisible && IsRectContains(x, y, mw.Left, mw.Top, mw.Width, mw.Height))  
  56.                 {  
  57.                     if (!mw.Enabled) return;  
  58.    
  59.                     if ((state & MouseState.Move) > 0)  
  60.                         _dispatcher.BeginInvoke(new MouseInputEventHandler(mw.OnMouseMove), mw, new MouseEventArgs(button, x - mw.Left, y - mw.Top));  
  61.                     if ((state & MouseState.Down) > 0)  
  62.                         _dispatcher.BeginInvoke(new MouseInputEventHandler(mw.OnMouseDown), mw, new MouseEventArgs(button, x - mw.Left, y - mw.Top));  
  63.                     if ((state & MouseState.Up) > 0)  
  64.                         _dispatcher.BeginInvoke(new MouseInputEventHandler(mw.OnMouseUp), mw, new MouseEventArgs(button, x - mw.Left, y - mw.Top));  
  65.                     if ((state & MouseState.Click) > 0)  
  66.                         _dispatcher.BeginInvoke(new MouseInputEventHandler(mw.OnMouseClick), mw, new MouseEventArgs(button, x - mw.Left, y - mw.Top));  
  67.                     if ((state & MouseState.Event) > 0)  
  68.                         _dispatcher.BeginInvoke(new MouseInputEventHandler(mw.OnMouseEvent), mw, new MouseEventArgs(button, x - mw.Left, y - mw.Top));  
  69.                     return;  
  70.                 }  
  71.             }  
  72.             catch (Exception e)  
  73.             {  
  74.                 throw new Exception(e.Message.ToString(), e);  
  75.             }  
  76.         }  
  77.     }  

  用户程序的窗体类只要派生于YFWindowBase类,就可以直接支持鼠标和按键功能了。用户代码如下:
   

 
  1. //主窗体  
  2.    internal sealed class MFWindow :YFWindowBase  
  3.    {  
  4.        public YFLabel label1;  
  5.        YFButton button1, button2, button3, button4, button5;  
  6.        public MFWindow()  
  7.        {  
  8.            //标签  
  9.            label1 = new YFLabel("就绪", 0, Height - 25, Width, 25);  
  10.            label1.TextAlign = TextAlignment.Left;  
  11.            label1.BackColor = ColorUtility.ColorFromRGB(189, 235, 255);  
  12.            label1.BorderStyle = BorderStyle.FixedSingle;  
  13.            //添加按钮  
  14.            button1 = new YFButton("触摸屏校准", 30, 35, 90, 40);  
  15.            button1.MouseClick += new MouseInputEventHandler(button_MouseClick);  
  16.            button2 = new YFButton("计算器",200, 35, 90, 40);  
  17.            button2.MouseClick += new MouseInputEventHandler(button_MouseClick);  
  18.            button3 = new YFButton("简易记事本", 30, 135, 90, 40);  
  19.            button3.MouseClick += new MouseInputEventHandler(button_MouseClick);  
  20.            button4 = new YFButton("关于...", 200, 135, 90, 40);  
  21.            button4.MouseClick += new MouseInputEventHandler(button_MouseClick);  
  22.            button5 = new YFButton("主菜单", 125, 85, 70, 40);  
  23.            button5.MouseClick += new MouseInputEventHandler(button_MouseClick);  
  24.            Children.Add(button1);  
  25.            Children.Add(button2);  
  26.            Children.Add(button3);  
  27.            Children.Add(button4);  
  28.            Children.Add(button5);  
  29.            Children.Add(label1);  
  30.            button3.Enable = false;  
  31.            //button3.Visible = false;  
  32.            //设置菜单  
  33.            Menu.AddItem(new MenuItem("触摸屏校准"));  
  34.            Menu.AddItem(new MenuItem("-"));  
  35.            Menu.AddItem(new MenuItem("计算器"));  
  36.            Menu.AddItem(new MenuItem("简易记事本"));  
  37.            Menu.AddItem(new MenuItem("-"));  
  38.            Menu.AddItem(new MenuItem("关于..."));  
  39.            Menu[3].Enabled = false;  
  40.            //Menu[3].Visible = false;  
  41.        }          
  42.        //按钮事件  
  43.        void button_MouseClick(object sender, MouseEventArgs e)  
  44.        {  
  45.            YFButton button=((YFButton)sender);  
  46.            switch (button.Text)  
  47.            {  
  48.                case "主菜单":  
  49.                    //弹出菜单  
  50.                    this.Menu.Show();  
  51.                    break;  
  52.                default:  
  53.                    OnMenuClick(new MenuEventArgs(0, button.Text));  
  54.                    break;  
  55.            }                            
  56.        }  

运行后的界面如下:
 

图1:主界面(按钮即支持鼠标也可以用按键切换输入焦点(right键等同于PC平台上的Tab键),并用OK键触发按键事件)

 

图2:单击主菜单按钮或单击“Menu”就可以弹出主菜单


 

图3:不要小看了上图的蓝色小圆,是我费了好大劲才绘制出来的(目前MF仅支持矩形框的填充)。
 

 
  1. private void DrawCircle(Color c,int x, int y, int r, DrawingContext dc)  
  2.     {  
  3.         Pen p=new Pen(c);  
  4.         SolidColorBrush b = new SolidColorBrush(c);  
  5.         int Offset45=(int)(0.707*r);  
  6.         int Offset30 = r / 2;  
  7.         int Offset60 = (int)(0.866 * r);  
  8.    
  9.         for (int i = Offset45; i < r + 1; i++)  
  10.         {  
  11.             dc.DrawEllipse(null, p, x, y, i, i);  
  12.         }  
  13.         dc.DrawRectangle(b, null, x - Offset45, y - Offset45,Offset45*2, Offset45*2);  
  14.         dc.DrawRectangle(b, null, x - Offset60, y - Offset30, Offset60 * 2, Offset30 * 2);  
  15.         dc.DrawRectangle(b, null, x - Offset30, y - Offset60, Offset30 * 2, Offset60 * 2);  
  16.    
  17.         dc.DrawLine(p, x - Offset60, y - Offset30, x - Offset30, y - Offset60);  
  18.         dc.DrawLine(p, x+ Offset60 , y + Offset30, x + Offset30 ,y + Offset60 );  
  19.         dc.DrawLine(p, x - Offset60, y + Offset30, x - Offset30 , y + Offset60);  
  20.         dc.DrawLine(p, x + Offset60, y - Offset30, x + Offset30, y - Offset60);  
  21.    }  
  22.    
  23.    

图5:终于在MF上实现了计算器的功能,目前不仅支持鼠标,也可以用光标键和OK键进行输入计算了。
别小看了计算器程序,由于MF仅有数字转字符串功能,没有实现字符串转数字的功能,我自己自定义了一些函数,用了一些特殊的用法才编写完成。
 

 
  1. //数字输入的部分代码  
  2. if (strInput == "0.") dblInput = 0;  
  3.         if (strInput != "0.")  
  4.         {  
  5.             strInput += b.Text;  
  6.             if (strInput.IndexOf('.')<1)  
  7.             {  
  8.                 dblInput *= 10;  
  9.                 dblInput += ToDouble(b.Text);  
  10.             }  
  11.             else 
  12.             {  
  13.                 int index = strInput.Length - strInput.LastIndexOf('.') - 1;  
  14.                 dblInput += ToDouble(b.Text) /System.Math.Pow(10,index);  
  15.             }  
  16.         }  
  17.         else if (b.Text != "0")  
  18.         {  
  19.             strInput = b.Text;  
  20.             dblInput = ToDouble(b.Text);  
  21.      }  

   下面是窗体的界面搭建部分,你绝对想不到用这么短的代码就实现了上图的界面布局(看过我以前.Net Micro Framework研究的网友,应该对MF本身提供的控件有印象,正是因为系统的控件不好用,我专门重新写了一套MF控件类)。


   

 
  1.  YFButton[] button=new YFButton[20];  
  2.     YFLabel lblInput = null;  
  3.     string[] strText = new string[] { "7""8""9""/""CE""4""5""6""*""%""1""2""3""-""1/x""0""+/-"".""+""=" };  
  4.     public YFCalc(string Title,int Width,int Height,YFWindowBase Parent)  
  5.         : base(Title, Width, Height, Parent)  
  6.     {  
  7.         int x=ClientRect.X,y=ClientRect.Y;  
  8.    
  9.         lblInput = new YFLabel("0.", x + 5, y + 5, ClientRect.Width - 10, 20);  
  10.         lblInput.TextAlign = TextAlignment.Right;  
  11.         Children.Add(lblInput);   
  12.         for (int i = 0; i < 20; i++)  
  13.         {  
  14.             if(i % 5==0 && i!=0)  
  15.             {  
  16.                 x = ClientRect.X;  
  17.                 y = y + 32;  
  18.             }  
  19.             button[i] = new YFButton("", x + 5, y + 32, 32, 28);  
  20.             button[i].Text = strText[i];  
  21.             button[i].MouseClick += new MouseInputEventHandler(button_MouseClick);   
  22.             x += 37;  
  23.             Children.Add(button[i]);   
  24.         }                 
  25.     }  
  26.     //按钮单击  
  27.     void button_MouseClick(object sender, MouseEventArgs e)  
  28. {  
  29. }  

其实这段时间以来,我一直在研究MF,虽然目前它还不是很成熟,但是随着研究的深入,越来越对它痴迷,越来越发现很多MF的宝藏(如果你有耐心的话,一定要看看MF底层框架的源码(通过反编译工具Reflector),你会发现很多很有意思的功能)。
MF相对于Windows XP/Vista、Windows CE而言,还只能算一个婴孩,但就是这样,就如一个伟人所说:孩子就是未来的希望。所以有理由相信MF的明天会更好









本文转自yefanqiu51CTO博客,原文链接:http://blog.51cto.com/yfsoft/322900,如需转载请自行联系原作者

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/392872.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

leetcode 344. 反转字符串

编写一个函数&#xff0c;其作用是将输入的字符串反转过来。输入字符串以字符数组 char[] 的形式给出。 不要给另外的数组分配额外的空间&#xff0c;你必须原地修改输入数组、使用 O(1) 的额外空间解决这一问题。 你可以假设数组中的所有字符都是 ASCII 码表中的可打印字符。…

事件捕获(capture)和冒泡事件(Bubble)

PS&#xff1a;这里是我从别人的博客中学习事件捕获和冒泡是的总结&#xff0c;如果你也感兴趣的话&#xff0c;建议你点击链接查看原博客的内容&#xff0c;他们写的都是很经典&#xff01; 对“捕获”和“冒泡”这两个概念&#xff0c;我想我们对冒泡更熟悉一些&…

gulp编译css_如何用gulp缩小CSS

gulp编译cssby Vinicius Gularte由Vinicius Gularte 如何用gulp缩小CSS (How to minify your CSS with gulp) In this article, Im going to show a simple way to automatically minify your CSS files using gulp. ?在本文中&#xff0c;我将展示一种使用gulp自动缩小CSS文…

线段树(区间更改,区间查最值)模板

线段树(区间更改,区间查最值)模板 主要重在理解线段树,理解了怎么改都可以,还有以后不要直接抄模板,要写出自己想的一份代码 &代码&#xff1a; #include <cstdio> #include <bitset> #include <iostream> #include <set> #include <cmath>…

Unity3D项目开发一点经验

我们主要使用3dsmax2010进行制作&#xff0c;输出FBX的类型导入Unity3D中。默认情况下&#xff0c;3dsmax8可以和U3D软件直接融合&#xff0c;自动转换为FBX物体。 注意事项如下&#xff1a; 1.面数控制 在MAX软件中制作单一GameObject物体的面数不能超过65000个三角形&#xf…

leetcode 142. 环形链表 II(set/快慢指针)

给定一个链表&#xff0c;返回链表开始入环的第一个节点。 如果链表无环&#xff0c;则返回 null。 为了表示给定链表中的环&#xff0c;我们使用整数 pos 来表示链表尾连接到链表中的位置&#xff08;索引从 0 开始&#xff09;。 如果 pos 是 -1&#xff0c;则在该链表中没有…

html5 支持表格吗,html5 – 在HTML 5中使用表格很好吗?

简单规则 – 使用表格表格数据&#xff0c;使用其他元素进行演示(使用CSS设计布局)&#xff0c;如div&#xff0c;section&#xff0c;aside&#xff0c;nav等。这为他们所持有的内容提供了意义&#xff0c;而不是为所有内容使用表事实是&#xff0c;开发人员在90年代使用了表格…

css网格_我如何记住CSS网格属性

css网格The syntax for CSS Grid is foreign and hard to remember. But if you can’t remember CSS Grid’s syntax, you won’t be confident when you use CSS Grid.CSS Grid的语法是外来的&#xff0c;很难记住。 但是&#xff0c;如果您不记得CSS Grid的语法&#xff0c;…

2017年读书计划(一)

前言 这篇博文就暂时不记录技术了&#xff0c;记录下生活。对自己今年2017年做个读书计划安排。 最近在看一部网络剧 - 《花间提壶方大厨》&#xff0c;也许你们会感觉我很无聊&#xff0c;我也是被头条带坏了&#xff0c;每天上班一个小时的地下交通-地铁&#xff0c;就借助上…

.net10个必备工具

1.NUnit 编写单元测试的工具2.NDoc 自动生成代码文档的工具3.NAnt 编译解决方案的工具4.CodeSmith 自动生成代码的工具5.FxCop 检查你的代码是否按照规范编写的工具6.Snippet Compiler 编译少量代码的工具7.ASP.NET Version Switcher Visual Studio .NET Project Conve…

音标

音标 oror ds念子音&#xff0c;ts念s音

leetcode 530. 二叉搜索树的最小绝对差(中序遍历)

给你一棵所有节点为非负值的二叉搜索树&#xff0c;请你计算树中任意两节点的差的绝对值的最小值。示例&#xff1a;输入&#xff1a;1\3/2输出&#xff1a; 1解释&#xff1a; 最小绝对差为 1&#xff0c;其中 2 和 1 的差的绝对值为 1&#xff08;或者 2 和 3&#xff09;。代…

计算机排线知识,一种计算机排线梳理装置制造方法及图纸

【技术实现步骤摘要】一种计算机排线梳理装置本技术涉及计算机排线梳理&#xff0c;具体涉及一种计算机排线梳理装置。技术介绍计算机俗称电脑&#xff0c;是现代一种用于高速计算的电子计算机器&#xff0c;可以进行数值计算&#xff0c;又可以进行逻辑计算&#xff0c;还具有…

github和pypi_如何将GitHub用作PyPi服务器

github和pypiI was looking for a hosted private PyPi Python Package server, that used credentials that the team already has (such as GitHub).我正在寻找一个托管的私有PyPi Python Package服务器&#xff0c;该服务器使用了团队已经拥有的凭据(例如GitHub)。 I didn’…

数据结构与算法---查找算法(Search Algorithm)

查找算法介绍 在java中&#xff0c;我们常用的查找有四种: 顺序(线性)查找 二分查找/折半查找 插值查找斐波那契查找1)线性查找算法 示例&#xff1a; 有一个数列&#xff1a; {1,8, 10, 89, 1000, 1234} &#xff0c;判断数列中是否包含此名称【顺序查找】 要求: 如果找到了&a…

Exchange Server 2007邮箱存储服务器的集群和高可用性技术(上)

高可用性矩阵-->见下图:邮箱服务器高可用性目标: 数据可用性-->保护邮箱数据免于失败和损坏服务可用性-->提高群集实效转移操作 简化群集管理 支持地理分散的群集 支持低成本大邮箱(GB)使用户可以基于业务需要更好的选择容错方案提高解决方案的可用性使用解决方案可…

【C/C++开发】C++实现字符串替换的两种方法

替换字符串replace() erase()//C 第一种替换字符串的方法用replace()|C 第二种替换字符串的方法用erase()和insert()【 Cstring|C replace()|C erase()|C insert()|C自定义替换字符串函数】#include<string> #include<iostream> using namespace std;//第一种替换字…

html设置按钮样式变为椭圆,css border-radius圆形变为椭圆形,位置:绝对

我正在围绕字体真棒图标创建一个圆圈。我的问题是&#xff0c;当我添加position: absolute圆成为一个椭圆。css border-radius圆形变为椭圆形&#xff0c;位置&#xff1a;绝对同样的情况&#xff0c;如果我是设置display: block这里是什么&#xff0c;我想实现的图像 -CONRADU…

《火球——UML大战需求分析》(第1章 大话UML)——1.5 小结和练习

说明&#xff1a; 《火球——UML大战需求分析》是我撰写的一本关于需求分析及UML方面的书&#xff0c;我将会在CSDN上为大家分享前面几章的内容&#xff0c;总字数在几万以上&#xff0c;图片有数十张。欢迎你按文章的序号顺序阅读&#xff0c;谢谢&#xff01;本书已经在各大网…

金陵科技学院计算机开设课程,金陵科技学院各专业介绍

各专业介绍会计学专业(四年制本科) 金融学专业(四年制本科)财务管理专业(四年制本科) 国际经济与贸易专业(四年制本科)市场营销专业(四年制本科)国际商务专业(三年制专科)物流管理专业(三年制专科) 对外汉语专业(四年制本科)古典文献(古籍修复)专业(四年制本科)行政管理(高级秘…