赞
踩
在毕业设计中做了一个基于AHRS的设计,涉及了姿态融合算法,为了调试算法参数性能,需要用到姿态显示上位机用来显示验证算法的效果。在收集了很多资料后,决定自己做一个三维姿态显示上位机。由于本人学识浅薄,程序中可能有很多目前没有发现的BUG,如有纰漏,敬请指正。
下面介绍一下我开发的大致方法:
1、HID数据传输,在开发的过程中,采用无线传输的方式将单片机的数据传输到电脑的USB端,再采用HID协议将USB端点的数据上传到电脑。
2、在电脑上位机中检测上传到电脑的数据,当检测到数据后,通过TCP协议与Unity3D通讯,Unity3D作为服务器,根据通信内容执行相关操作。
3、上位机UI采用开源的PanuonUI进行开发。
该上位机的优劣:
优点:开发简单,不用使用OpenGL开发,仅使用C#语言就可以完成开发;界面美观,贴图和背景相比OpenGL开发效果好很多;运行中Unity3D使用了GPU渲染,相比软渲染流畅度更好。
缺点:Unity3D的EXE文件较大,比采用C++开发的上位机体量大很多;可以作为学习和实验使用,距离商用还有很大差距。
由于整个上位机的开发过程叙述起来比较麻烦,这里直接放出源码供需要的人下载使用。
HID对接收数据进行转换:
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; using System.Windows; using System.Windows.Controls; using System.Windows.Data; using System.Windows.Documents; using System.Windows.Input; using System.Windows.Media; using System.Windows.Media.Imaging; using System.Windows.Navigation; using System.Windows.Shapes; using System.Collections.ObjectModel; using System.Windows.Interop; using System.Windows.Threading; using HID; using System.Runtime.InteropServices; using Panuon.UI.Silver; namespace CubeAttitudeShow { public partial class MainWindow : WindowX { #region HID_协议定义与实现 /*定义HID数传的内容*/ public Hid myHid = new Hid(); public IntPtr myHidPtr = new IntPtr(); string IMUKeyNum; int temp; public bool BiaoDingFlag = false; public bool BiaoDingFlag2 = false; double MagX,MagY,MagZ; string SmagX, SmagY, SmagZ; public string Dis1; public string Dis2; public string Yaw; public string Pitch; public string Roll; /***************************************************************************/ /// <summary> /// 数据到达事件 /// </summary> /// <param name="sender"></param> /// <param name="e"></param> protected void myhid_DataReceived(object sender, report e) { if (lianjieflag) { RecDataBuffer = e.reportBuff; for (int a = 0; a < 6; a++) TransformData[a] = RecDataBuffer[a + 5]; WpfServer.SendMessage(TransformData); /*把数据显示到列表框*/ IMUKeyNum = RecDataBuffer[0].ToString();//按键信息 temp = 0; if (RecDataBuffer[1] >= 0x80) //判定是否是负数 temp = 0 - ((RecDataBuffer[1] & 0x7f) * 256 + RecDataBuffer[2]); else temp = (RecDataBuffer[1]) * 256 + RecDataBuffer[2]; Dis1 = temp.ToString();//基站 1 距离 temp = 0; if (RecDataBuffer[3] >= 0x80) //判定是否是负数 temp = 0 - ((RecDataBuffer[3] & 0x7f) * 256 + RecDataBuffer[4]); else temp = (RecDataBuffer[3]) * 256 + RecDataBuffer[4]; Dis2 = temp.ToString();//基站 2 距离 temp = 0; if (RecDataBuffer[5] >= 0x80) temp = 0 - ((RecDataBuffer[5] & 0x7f) * 256 + RecDataBuffer[6]) / 10; else temp = ((RecDataBuffer[5]) * 256 + (RecDataBuffer[6])) / 10; Yaw = temp.ToString();//偏航角 MagX = temp * 0.00256; SmagX = MagX.ToString(); temp = 0; if (RecDataBuffer[7] >= 0x80) temp = 0 - ((RecDataBuffer[7] & 0x7f) * 256 + RecDataBuffer[8]) / 10; else temp = ((RecDataBuffer[7]) * 256 + (RecDataBuffer[8])) / 10; Pitch = temp.ToString();//俯仰角 MagY = temp * 0.00256; SmagY = MagY.ToString(); temp = 0; if (RecDataBuffer[9] >= 0x80) temp = 0 - ((RecDataBuffer[9] & 0x7f) * 256 + RecDataBuffer[10]) / 10; else temp = ((RecDataBuffer[9]) * 256 + (RecDataBuffer[10])) / 10; Roll = temp.ToString();//滚转角 MagZ = temp * 0.00256; SmagZ = MagZ.ToString(); this.Dispatcher.Invoke(new Action(delegate { KeyCommand.Text = IMUKeyNum; })); this.Dispatcher.Invoke(new Action(delegate { Station1.Text = Dis1; })); this.Dispatcher.Invoke(new Action(delegate { Station2.Text = Dis2; })); this.Dispatcher.Invoke(new Action(delegate { PitchAngle.Text = Pitch; })); this.Dispatcher.Invoke(new Action(delegate { YawAngle.Text = Yaw; })); this.Dispatcher.Invoke(new Action(delegate { RollAngle.Text = Roll; })); } } /// <summary> /// 设备移除事件 /// </summary> /// <param name="sender"></param> /// <param name="e"></param> protected void myhid_DeviceRemoved(object sender, EventArgs e) { CloseDevice(); this.Dispatcher.Invoke(new Action(delegate { LianJie.Content = tubiao + " 失去连接 "; })); lianjieflag = false; } /***************************************************************************/ Byte[] RecDataBuffer = new byte[90]; /// <summary> /// 打开设备 /// </summary> protected bool OpenDevice() { if (myHid.Opened == false) { UInt16 myVendorID = 0x3333; UInt16 myProductID = 0x4444; if ((int)(myHidPtr = myHid.OpenDevice(myVendorID, myProductID)) != -1)//调用开启HID函数来开启HID口 { return true; } else { return false; } } else { return true; } } /// <summary> /// 设备关闭 /// </summary> protected void CloseDevice() { myHid.CloseDevice(myHidPtr); } /// <summary> /// 窗口加载时完成事件说明 /// </summary> protected void HID_Loaded() { myHid.DataReceived += new EventHandler<HID.report>(myhid_DataReceived); //订阅DataRec事件 myHid.DeviceRemoved += new EventHandler(myhid_DeviceRemoved); } #endregion } }
TCP客户端:
using System.Net.Sockets; using System.Net; using System.Threading; using System.Diagnostics; using System.Text; using System; namespace CubeAttitudeShow { class TcpServer { //私有成员 private static byte[] result = new byte[1024]; private int myProt = 500; //端口 static Socket serverSocket;//服务器接口 static Socket clientSocket;//客户接口 Thread myThread; static Thread receiveThread; //属性 public int port { get; set; } //方法 internal void StartServer() { //服务器IP地址 IPAddress ip = IPAddress.Parse("127.0.0.1"); serverSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp); serverSocket.Bind(new IPEndPoint(ip, myProt)); //绑定IP地址:端口 serverSocket.Listen(10); //设定最多10个排队连接请求 myThread = new Thread(ListenClientConnect); myThread.Start(); } internal void QuitServer() { serverSocket.Close(); clientSocket.Close(); myThread.Abort(); receiveThread.Abort(); } internal void SendMessage(byte[] msg) { clientSocket.Send(msg); } /// <summary> /// 监听客户端连接 /// </summary> private static void ListenClientConnect() { while (true) { try { clientSocket = serverSocket.Accept(); //clientSocket.Send(Encoding.ASCII.GetBytes("Server Say Hello")); receiveThread = new Thread(ReceiveMessage); receiveThread.Start(clientSocket); } catch (Exception) { } } } /// <summary> /// 接收消息 /// </summary> /// <param name="clientSocket"></param> private static void ReceiveMessage(object clientSocket) { Socket myClientSocket = (Socket)clientSocket; while (true) { try { //通过clientSocket接收数据 int receiveNumber = myClientSocket.Receive(result); //Debug.WriteLine("接收客户端{0}消息{1}", myClientSocket.RemoteEndPoint.ToString(), Encoding.ASCII.GetString(result, 0, receiveNumber)); } catch (Exception ex) { try { Debug.WriteLine(ex.Message); myClientSocket.Shutdown(SocketShutdown.Both); myClientSocket.Close(); break; } catch (Exception) { } } } } } }
Unity嵌套:
using System; using System.Windows.Forms; using System.Runtime.InteropServices; using System.Diagnostics; using System.Threading; namespace CubeAttitudeShow { public partial class UnityControl : UserControl { [DllImport("User32.dll")] static extern bool MoveWindow(IntPtr handle, int x, int y, int width, int height, bool redraw); internal delegate int WindowEnumProc(IntPtr hwnd, IntPtr lparam); [DllImport("user32.dll")] internal static extern bool EnumChildWindows(IntPtr hwnd, WindowEnumProc func, IntPtr lParam); [DllImport("user32.dll")] static extern int SendMessage(IntPtr hWnd, int msg, IntPtr wParam, IntPtr lParam); private Process process; private IntPtr unityHWND = IntPtr.Zero; private const int WM_ACTIVATE = 0x0006; private readonly IntPtr WA_ACTIVE = new IntPtr(1); private readonly IntPtr WA_INACTIVE = new IntPtr(0); public UnityControl() { InitializeComponent(); this.Load += UnityControl_Load; panel1.Resize += panel1_Resize; } private void UnityControl_Load(object sender, EventArgs e) { try { process = new Process(); //process.StartInfo.FileName = @"D:\毕业设计\毕业设计\电脑软件设计\CubeAttitude\CubeAttitude3DShow\EXE\CubeAttitude3DShow.exe"; process.StartInfo.FileName = Application.StartupPath + @"\Cube Attitude Exe\CubeAttitude3DShow.exe"; process.StartInfo.Arguments = "-parentHWND " + panel1.Handle.ToInt32() + " " + Environment.CommandLine; process.StartInfo.UseShellExecute = true; process.StartInfo.CreateNoWindow = true; process.Start(); process.WaitForInputIdle(); // Doesn't work for some reason ?! //unityHWND = process.MainWindowHandle; EnumChildWindows(panel1.Handle, WindowEnum, IntPtr.Zero); //unityHWNDLabel.Text = "Unity HWND: 0x" + unityHWND.ToString("X8"); } catch (Exception ex) { //unityHWNDLabel.Text = ex.Message; //MessageBox.Show(ex.Message); } } internal void ActivateUnityWindow() { SendMessage(unityHWND, WM_ACTIVATE, WA_ACTIVE, IntPtr.Zero); } internal void DeactivateUnityWindow() { SendMessage(unityHWND, WM_ACTIVATE, WA_INACTIVE, IntPtr.Zero); } private int WindowEnum(IntPtr hwnd, IntPtr lparam) { unityHWND = hwnd; ActivateUnityWindow(); return 0; } private void panel1_Resize(object sender, EventArgs e) { MoveWindow(unityHWND, 0, 0, panel1.Width, panel1.Height, true); ActivateUnityWindow(); } // Close Unity application internal void Form1_FormClosed() { try { process.CloseMainWindow(); Thread.Sleep(1000); while (process.HasExited == false) process.Kill(); } catch (Exception) { } } internal void Form1_Activated() { ActivateUnityWindow(); } internal void Form1_Deactivate() { DeactivateUnityWindow(); } } }
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。