`
isiqi
  • 浏览: 16500193 次
  • 性别: Icon_minigender_1
  • 来自: 济南
社区版块
存档分类
最新评论

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

阅读更多

基于MF系统的Windows SideShow界面是非常炫的(如下图)。既然微软能用.Net Micro Framework实现这么棒的界面效果,我想我们也能做到。
(SideShow模拟器界面和游戏程序中的右键菜单—注意菜单弹出后,其它的界面变暗了)
现在的任务是设计一套支持鼠标(或触摸屏)的窗体框架(目前MF提供的Window类仅支持按键功能),所以正好把SideShow如此炫的界面元素也可以添加进来。
用过MF的人知道是用下面的方法来实现按键事件接收的,既然我们要支持鼠标功能,所以最好也用类似的机理实现。
//按键事件
protected override void OnButtonDown(ButtonEventArgs e)
{
switch (e.Button)
{
//按下确定键
case Button.Select:
break;
//按下左键
case Button.Left:
break;
//按下右键
case Button.Right:
break;
//按向上
case Button.Up:
break;
//按向下
case Button.Down:
break;
//按下菜单
case Button.Menu:
break;
//按下返回键
case Button.Back:
break;
}
base.OnButtonDown(e);
}
用反编译工具仔细研究了MF底层库代码(.Net FrameWork 太庞大了,一个人绝对短时间内看不完,其实也很难看下去,但是对刚刚起步的MF来说,.Net Micro FrameWork就简单多了),终于理清了头绪。主要原理是在鼠标信息处理线程中通过Application.Current.Windows 属性(该属性存放了当前实例所有派生于Window类的窗体)和应用实例的this.Dispatcher属性的BeginInvoke方法,外部调用窗体鼠标事件函数。充分利用基类虚函数的妙处来实现类似按键信息处理的功能。
YFWindowBase类中声明如下虚拟鼠标事件函数。
//鼠标移动
public virtual void OnMouseMove(object sender, MouseEventArgs e)
{
if (MouseMove != null) MouseMove(sender, e);
}
//鼠标单击
public virtual void OnMouseClick(object sender, MouseEventArgs e)
{
if (MouseClick != null) MouseClick(sender, e);
}
//按下
public virtual void OnMouseDown(object sender, MouseEventArgs e)
{
if (MouseDown != null) MouseDown(sender, e);
}
//抬起
public virtual void OnMouseUp(object sender, MouseEventArgs e)
{
if (MouseUp != null) MouseUp(sender, e);
}
在鼠标信息处理函数中执行如下的代码即可。
//处理鼠标消息
private static void TransactMouse(MouseState state, int x, int y, MouseButtons button)
{
if (Application.Current == null) return;
for (int i = Application.Current.Windows.Count - 1; i >= 0; i--)
{
try
{
YFWindowBase mw = Application.Current.Windows[i] as YFWindowBase;
if (mw.Enabled && mw.IsVisible)
{
//判断子窗体
bool bReturn = false;
for (int j = mw.Children.Count - 1; j >= 0; j--)
{
//仅最上层并且可视的控件接收鼠标消息
YFControl cl = mw.Children[j];
if (!bReturn && cl.Visible && IsRectContains(x, y, mw.Left + cl.Left, mw.Top + cl.Top, cl.Width, cl.Height))
{
if (cl.Enable)//Enable和Visible不一样,Enable即使无效,下层控件也没有机会获得鼠标消息
{
if (!cl._EnterFlag)
{
cl._EnterFlag = true;
_dispatcher.BeginInvoke(new MouseInputEventHandler(cl.OnMouseEnter), cl, new MouseEventArgs(button, x - cl.Left - mw.Left, y - cl.Top - mw.Top));
}
if ((state & MouseState.Move) > 0)
_dispatcher.BeginInvoke(new MouseInputEventHandler(cl.OnMouseMove), cl, new MouseEventArgs(button, x - cl.Left - mw.Left, y - cl.Top - mw.Top));
if ((state & MouseState.Down) > 0)
_dispatcher.BeginInvoke(new MouseInputEventHandler(cl.OnMouseDown), cl, new MouseEventArgs(button, x - cl.Left - mw.Left, y - cl.Top - mw.Top));
if ((state & MouseState.Up) > 0)
_dispatcher.BeginInvoke(new MouseInputEventHandler(cl.OnMouseUp), cl, new MouseEventArgs(button, x - cl.Left - mw.Left, y - cl.Top - mw.Top));
if ((state & MouseState.Click) > 0)
_dispatcher.BeginInvoke(new MouseInputEventHandler(cl.OnMouseClick), cl, new MouseEventArgs(button, x - cl.Left - mw.Left, y - cl.Top - mw.Top));
}
//向主窗体传OnMouseEvent消息,为了绘制鼠标
if ((state & MouseState.Event) > 0)
_dispatcher.BeginInvoke(new MouseInputEventHandler(mw.OnMouseEvent), mw, new MouseEventArgs(button, x - mw.Left, y - mw.Top));
bReturn = true;
}
else
{
if (cl._EnterFlag)
{
cl._EnterFlag = false;
_dispatcher.BeginInvoke(new MouseInputEventHandler(cl.OnMouseLeave), cl, new MouseEventArgs(button, x - cl.Left, y - cl.Top));
}
}
}
if (bReturn) return;
}
//仅最上层并且可视的窗体接收鼠标消息
if (mw.IsVisible && IsRectContains(x, y, mw.Left, mw.Top, mw.Width, mw.Height))
{
if (!mw.Enabled) return;
if ((state & MouseState.Move) > 0)
_dispatcher.BeginInvoke(new MouseInputEventHandler(mw.OnMouseMove), mw, new MouseEventArgs(button, x - mw.Left, y - mw.Top));
if ((state & MouseState.Down) > 0)
_dispatcher.BeginInvoke(new MouseInputEventHandler(mw.OnMouseDown), mw, new MouseEventArgs(button, x - mw.Left, y - mw.Top));
if ((state & MouseState.Up) > 0)
_dispatcher.BeginInvoke(new MouseInputEventHandler(mw.OnMouseUp), mw, new MouseEventArgs(button, x - mw.Left, y - mw.Top));
if ((state & MouseState.Click) > 0)
_dispatcher.BeginInvoke(new MouseInputEventHandler(mw.OnMouseClick), mw, new MouseEventArgs(button, x - mw.Left, y - mw.Top));
if ((state & MouseState.Event) > 0)
_dispatcher.BeginInvoke(new MouseInputEventHandler(mw.OnMouseEvent), mw, new MouseEventArgs(button, x - mw.Left, y - mw.Top));
return;
}
}
catch (Exception e)
{
throw new Exception(e.Message.ToString(), e);
}
}
}
用户程序的窗体类只要派生于YFWindowBase类,就可以直接支持鼠标和按键功能了。用户代码如下:
//主窗体
internal sealed class MFWindow :YFWindowBase
{
public YFLabel label1;
YFButton button1, button2, button3, button4, button5;
public MFWindow()
{
//标签
label1 = new YFLabel("就绪", 0, Height - 25, Width, 25);
label1.TextAlign = TextAlignment.Left;
label1.BackColor = ColorUtility.ColorFromRGB(189, 235, 255);
label1.BorderStyle = BorderStyle.FixedSingle;
//添加按钮
button1 = new YFButton("触摸屏校准", 30, 35, 90, 40);
button1.MouseClick += new MouseInputEventHandler(button_MouseClick);
button2 = new YFButton("计算器",200, 35, 90, 40);
button2.MouseClick += new MouseInputEventHandler(button_MouseClick);
button3 = new YFButton("简易记事本", 30, 135, 90, 40);
button3.MouseClick += new MouseInputEventHandler(button_MouseClick);
button4 = new YFButton("关于...", 200, 135, 90, 40);
button4.MouseClick += new MouseInputEventHandler(button_MouseClick);
button5 = new YFButton("主菜单", 125, 85, 70, 40);
button5.MouseClick += new MouseInputEventHandler(button_MouseClick);
Children.Add(button1);
Children.Add(button2);
Children.Add(button3);
Children.Add(button4);
Children.Add(button5);
Children.Add(label1);
button3.Enable = false;
//button3.Visible = false;
//设置菜单
Menu.AddItem(new MenuItem("触摸屏校准"));
Menu.AddItem(new MenuItem("-"));
Menu.AddItem(new MenuItem("计算器"));
Menu.AddItem(new MenuItem("简易记事本"));
Menu.AddItem(new MenuItem("-"));
Menu.AddItem(new MenuItem("关于..."));
Menu[3].Enabled = false;
//Menu[3].Visible = false;
}
//按钮事件
void button_MouseClick(object sender, MouseEventArgs e)
{
YFButton button=((YFButton)sender);
switch (button.Text)
{
case "主菜单":
//弹出菜单
this.Menu.Show();
break;
default:
OnMenuClick(new MenuEventArgs(0, button.Text));
break;
}
}
运行后的界面如下:
1:主界面(按钮即支持鼠标也可以用按键切换输入焦点(right键等同于PC平台上的Tab键),并用OK键触发按键事件)
2:单击主菜单按钮或单击“Menu”就可以弹出主菜单
3:不要小看了上图的蓝色小圆,是我费了好大劲才绘制出来的(目前MF仅支持矩形框的填充)。
private void DrawCircle(Color c,int x, int y, int r, DrawingContext dc)
{
Penp=new Pen(c);
SolidColorBrush b = new SolidColorBrush(c);
int Offset45=(int)(0.707*r);
int Offset30 = r / 2;
int Offset60 = (int)(0.866 * r);
for (int i = Offset45; i < r + 1; i++)
{
dc.DrawEllipse(null, p, x, y, i, i);
}
dc.DrawRectangle(b, null, x - Offset45, y - Offset45,Offset45*2, Offset45*2);
dc.DrawRectangle(b, null, x - Offset60, y - Offset30, Offset60 * 2, Offset30 * 2);
dc.DrawRectangle(b, null, x - Offset30, y - Offset60, Offset30 * 2, Offset60 * 2);
dc.DrawLine(p, x - Offset60, y - Offset30, x - Offset30, y - Offset60);
dc.DrawLine(p, x+ Offset60 , y + Offset30, x + Offset30 ,y + Offset60 );
dc.DrawLine(p, x - Offset60, y + Offset30, x - Offset30 , y + Offset60);
dc.DrawLine(p, x + Offset60, y - Offset30, x + Offset30, y - Offset60);
}
5:终于在MF上实现了计算器的功能,目前不仅支持鼠标,也可以用光标键和OK键进行输入计算了。
别小看了计算器程序,由于MF仅有数字转字符串功能,没有实现字符串转数字的功能,我自己自定义了一些函数,用了一些特殊的用法才编写完成。
//数字输入的部分代码
if (strInput == "0.") dblInput = 0;
if (strInput != "0.")
{
strInput += b.Text;
if (strInput.IndexOf('.')<1)
{
dblInput *= 10;
dblInput += ToDouble(b.Text);
}
else
{
int index = strInput.Length - strInput.LastIndexOf('.') - 1;
dblInput += ToDouble(b.Text) /System.Math.Pow(10,index);
}
}
else if (b.Text != "0")
{
strInput = b.Text;
dblInput = ToDouble(b.Text);
}
下面是窗体的界面搭建部分,你绝对想不到用这么短的代码就实现了上图的界面布局(看过我以前.Net Micro Framework研究的网友,应该对MF本身提供的控件有印象,正是因为系统的控件不好用,我专门重新写了一套MF控件类)。
YFButton[] button=new YFButton[20];
YFLabel lblInput = null;
string[] strText = new string[] { "7", "8", "9", "/", "CE", "4", "5", "6", "*", "%", "1", "2", "3", "-", "1/x", "0", "+/-", ".", "+", "=" };
public YFCalc(string Title,int Width,int Height,YFWindowBase Parent)
: base(Title, Width, Height, Parent)
{
int x=ClientRect.X,y=ClientRect.Y;
lblInput = new YFLabel("0.", x + 5, y + 5, ClientRect.Width - 10, 20);
lblInput.TextAlign = TextAlignment.Right;
Children.Add(lblInput);
for (int i = 0; i < 20; i++)
{
if(i % 5==0 && i!=0)
{
x = ClientRect.X;
y = y + 32;
}
button[i] = new YFButton("", x + 5, y + 32, 32, 28);
button[i].Text = strText[i];
button[i].MouseClick += new MouseInputEventHandler(button_MouseClick);
x += 37;
Children.Add(button[i]);
}
}
//按钮单击
void button_MouseClick(object sender, MouseEventArgs e)
{
}
其实这段时间以来,我一直在研究MF,虽然目前它还不是很成熟,但是随着研究的深入,越来越对它痴迷,越来越发现很多MF的宝藏(如果你有耐心的话,一定要看看MF底层框架的源码(通过反编译工具Reflector),你会发现很多很有意思的功能)。
MF相对于Windows XP/Vista、Windows CE而言,还只能算一个婴孩,但就是这样,就如一个伟人所说:孩子就是未来的希望。所以有理由相信MF的明天会更好。
分享到:
评论

相关推荐

    飞思卡尔宣布推出面向Windows Vista SideShow应用的i.MXS开发工具包

    微软.NET Micro Framework业务主管Colin Miller对此表示支持,认为飞思卡尔的开发工具包将助力开发者为各种市场,包括客户、工业和医药领域,快速开发新的SideShow解决方案和其他基于.NET Micro Framework的嵌入式...

    Sideshow

    在IT行业中,"Sideshow"通常指的是Adobe公司开发的一种创新技术,它主要用于设计和创建动态图形,特别是与屏幕显示和交互相关的元素。Sideshow技术主要应用于数字标牌、移动设备和电视等平台,为用户提供丰富的视觉...

    Vista WeatherSideShow Code

    3. **C#编程语言**: 作为微软开发的面向对象的编程语言,C#是创建SideShow应用的理想选择,因为它提供了对.NET Framework的强大支持,使得创建用户界面和处理系统服务变得简单。 4. **SideShow模拟器**: 在实际设备...

    Vista Media Center SideShow Gadget x86

    Vista Media Center SideShow Gadget x86

    开源项目-sideshow-apns2.zip

    开源项目“sideshow-apns2”是一个用Go语言编写的工具,用于实现Apple的新型HTTP/2推送通知服务(APNs)协议。这个项目旨在帮助开发者更高效、更可靠地向iOS、iPadOS以及macOS设备发送推送通知。在本文中,我们将...

    Professional.Windows.Vista.Gadgets.Programming

    这部分内容详细讲解了如何使用.NET Framework、Ajax、ASP.NET、C#等技术来开发Sidebar小工具。开发者可以通过调用Sidebar小工具APIs来实现更复杂的功能,如数据获取、处理和展示。此外,书中还介绍了一些增强Sidebar...

    Wrox.Professional.Windows.Vista.Gadgets.Programming.Dec.2007.rar

    8. **案例研究**:可能包含多个实际项目案例,分析和展示了各种类型小部件的实现过程和最佳实践。 9. **安全性**:讨论小部件的安全风险和防止恶意代码的措施。 通过阅读这本书,开发者不仅能掌握Vista小部件和...

    Windows Vista Gadgets Programming Dec.2007.pdf

    Windows SideShow是Windows Vista中另一个创新功能,旨在为便携式设备提供扩展显示界面,即使在主设备处于睡眠或休眠模式时也能继续使用。SideShow Gadgets能够在这些设备上显示和更新信息,如电子邮件摘要、日历...

    Vista 控件全面的控件

    在Vista中,还引入了Windows SideShow,这是一种扩展的用户界面技术,允许外围设备(如智能手表或小屏幕显示器)显示与主计算机同步的信息。开发者可以利用SideShow API创建支持这些设备的应用,提供如邮件、日历和...

    Windows Vista 领航系列课程(6):开发基于RSS的应用程序

    - 使用.NET Framework 3.0或更高版本,开发者可以利用Windows Presentation Foundation (WPF) 和Windows Communication Foundation (WCF) 来构建RSS应用。 - WPF提供了丰富的UI设计工具,而WCF则用于服务间的通信...

    Windows Vista API 分类整理

    1. Aero Glass效果:Windows Vista引入了透明和半透明的Aero Glass界面,API如DWM(Desktop Window Manager)允许开发者在自己的应用程序中实现类似的效果。 2. Flip3D:这是一种任务切换器,API如...

    IOS推送通知测试环境

    苹果的推送通知服务(APNs,Apple Push Notification service)是实现这一功能的核心。开发者需要在Apple Developer账号中设置并获取用于推送的证书和密钥,这包括生产环境和开发环境的证书,因为推送通知需要在不同...

    Windows Vista 产品兼容性(4):新瓶装老酒,在现有应用程序中使用Vista新特性

    本资料包主要关注的是如何在现有的应用程序中充分利用Vista的新特性,以实现更好的性能和兼容性。下面我们将详细探讨这些关键知识点。 一、Aero Glass界面 Vista引入了Aero Glass透明效果,提供了一种全新的用户...

    teenpatti-parel

    在这个项目中,我们可以期待看到JavaScript的实现,这可能涉及到前端用户界面、游戏逻辑、网络通信等方面的知识点。 在JavaScript方面,以下是一些可能涉及的关键技术点: 1. **DOM操作**:项目可能使用JavaScript...

    Windows Vista API

    开发者可以利用API实现自定义窗口外观,提供更丰富的用户体验。 3. **Windows SideShow**:这是一种扩展显示技术,允许设备(如智能手表或移动电话)与主计算机进行通信,显示相关信息。开发者可以创建SideShow内容...

    Windows 7 小图标

    在IT领域,操作系统界面的美观性和个性化经常是用户关注的焦点之一。Windows 7作为一款深受用户喜爱的操作系统,其小图标设计独特,为用户提供了一种个性化桌面的方式。本篇文章将详细探讨Windows 7的小图标及其相关...

    joiner:使用“go generate”命令的通用 strings.Join 实现

    使用 Go 1.4 中的go generate命令的通用strings.Join实现。 有关请参阅。 试试这个例子 安装这个包: go install github.com/bslatkin/joiner 进入example目录: go generate go run *.go 你应该看到: My ...

Global site tag (gtag.js) - Google Analytics