mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-22 08:38:36 -04:00
9c836ab037
move mavlink structure/currentstate around for future mods update old firmware git hashs mod some error descriptions AP_mount camera trigger mod modify raw param display with units/range/desc add radio support for 868mhz update ch7 options updated dataflashlog format small df log parser mod for bad gps loc renable menu to always dock. right click for autohide
2407 lines
99 KiB
C#
2407 lines
99 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.ComponentModel;
|
|
using System.Configuration;
|
|
using System.Drawing;
|
|
using System.Text;
|
|
using System.Windows.Forms;
|
|
using System.IO;
|
|
using System.Xml;
|
|
using System.Collections;
|
|
using System.Net;
|
|
using System.Text.RegularExpressions;
|
|
using System.Diagnostics;
|
|
using System.Runtime.InteropServices;
|
|
using System.Speech.Synthesis;
|
|
using System.Globalization;
|
|
using System.Threading;
|
|
using System.Net.Sockets;
|
|
using ArdupilotMega.Utilities;
|
|
using IronPython.Hosting;
|
|
using log4net;
|
|
using ArdupilotMega.Controls;
|
|
using System.Security.Cryptography;
|
|
using ArdupilotMega.Comms;
|
|
using ArdupilotMega.Arduino;
|
|
using System.IO.Ports;
|
|
using Transitions;
|
|
|
|
namespace ArdupilotMega
|
|
{
|
|
public partial class MainV2 : Form
|
|
{
|
|
private static readonly ILog log =
|
|
LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
|
|
|
|
// used to hide/show console window
|
|
[DllImport("user32.dll")]
|
|
public static extern int FindWindow(string szClass, string szTitle);
|
|
[DllImport("user32.dll")]
|
|
public static extern int ShowWindow(int Handle, int showState);
|
|
|
|
const int SW_SHOWNORMAL = 1;
|
|
const int SW_HIDE = 0;
|
|
|
|
/// <summary>
|
|
/// Main Comport interface
|
|
/// </summary>
|
|
public static MAVLink comPort = new MAVLink();
|
|
/// <summary>
|
|
/// Comport name
|
|
/// </summary>
|
|
public static string comPortName = "";
|
|
/// <summary>
|
|
/// use to store all internal config
|
|
/// </summary>
|
|
public static Hashtable config = new Hashtable();
|
|
/// <summary>
|
|
/// used to prevent comport access for exclusive use
|
|
/// </summary>
|
|
public static bool giveComport = false;
|
|
/// <summary>
|
|
/// mono detection
|
|
/// </summary>
|
|
public static bool MONO = false;
|
|
/// <summary>
|
|
/// speech engine enable
|
|
/// </summary>
|
|
public static bool speechEnable = false;
|
|
/// <summary>
|
|
/// spech engine static class
|
|
/// </summary>
|
|
public static Speech speechEngine = null;
|
|
/// <summary>
|
|
/// joystick static class
|
|
/// </summary>
|
|
public static Joystick joystick = null;
|
|
/// <summary>
|
|
/// track last joystick packet sent. used to control rate
|
|
/// </summary>
|
|
DateTime lastjoystick = DateTime.Now;
|
|
/// <summary>
|
|
/// hud background image grabber from a video stream - not realy that efficent. ie no hardware overlays etc.
|
|
/// </summary>
|
|
public static WebCamService.Capture cam = null;
|
|
/// <summary>
|
|
/// controls the main serial reader thread
|
|
/// </summary>
|
|
bool serialThread = false;
|
|
/// <summary>
|
|
/// used for mini http server for websockets/mjpeg video stream, and network link kmls
|
|
/// </summary>
|
|
private TcpListener listener;
|
|
/// <summary>
|
|
/// track the last heartbeat sent
|
|
/// </summary>
|
|
private DateTime heatbeatSend = DateTime.Now;
|
|
/// <summary>
|
|
/// used to call anything as needed.
|
|
/// </summary>
|
|
public static MainV2 instance = null;
|
|
|
|
public static MainSwitcher View;
|
|
|
|
/// <summary>
|
|
/// used to feed in a network link kml to the http server
|
|
/// </summary>
|
|
public string georefkml = "";
|
|
|
|
/// <summary>
|
|
/// store the time we first connect
|
|
/// </summary>
|
|
DateTime connecttime = DateTime.Now;
|
|
|
|
/// <summary>
|
|
/// enum of firmwares
|
|
/// </summary>
|
|
public enum Firmwares
|
|
{
|
|
ArduPlane,
|
|
ArduCopter2,
|
|
ArduRover,
|
|
Ateryx
|
|
}
|
|
|
|
DateTime connectButtonUpdate = DateTime.Now;
|
|
/// <summary>
|
|
/// declared here if i want a "single" instance of the form
|
|
/// ie configuration gets reloaded on every click
|
|
/// </summary>
|
|
GCSViews.FlightData FlightData;
|
|
GCSViews.FlightPlanner FlightPlanner;
|
|
//GCSViews.ConfigurationView.Setup Configuration;
|
|
GCSViews.Simulation Simulation;
|
|
//GCSViews.Firmware Firmware;
|
|
//GCSViews.Terminal Terminal;
|
|
|
|
private Form connectionStatsForm;
|
|
private ConnectionStats _connectionStats;
|
|
|
|
/// <summary>
|
|
/// This 'Control' is the toolstrip control that holds the comport combo, baudrate combo etc
|
|
/// Otiginally seperate controls, each hosted in a toolstip sqaure, combined into this custom
|
|
/// control for layout reasons.
|
|
/// </summary>
|
|
static internal ConnectionControl _connectionControl;
|
|
|
|
public MainV2()
|
|
{
|
|
log.Info("Mainv2 ctor");
|
|
|
|
Form splash = Program.Splash;
|
|
|
|
string strVersion = System.Reflection.Assembly.GetExecutingAssembly().GetName().Version.ToString();
|
|
|
|
strVersion = "mav " + MAVLink.MAVLINK_WIRE_PROTOCOL_VERSION;
|
|
|
|
splash.Text = "Mission Planner " + Application.ProductVersion + " " + strVersion;
|
|
|
|
splash.Refresh();
|
|
|
|
Application.DoEvents();
|
|
|
|
instance = this;
|
|
|
|
InitializeComponent();
|
|
|
|
View = MyView;
|
|
|
|
_connectionControl = toolStripConnectionControl.ConnectionControl;
|
|
_connectionControl.CMB_baudrate.TextChanged += this.CMB_baudrate_TextChanged;
|
|
_connectionControl.CMB_baudrate.SelectedIndexChanged += this.CMB_baudrate_SelectedIndexChanged;
|
|
_connectionControl.CMB_serialport.SelectedIndexChanged += this.CMB_serialport_SelectedIndexChanged;
|
|
_connectionControl.CMB_serialport.Enter += this.CMB_serialport_Enter;
|
|
_connectionControl.CMB_serialport.Click += this.CMB_serialport_Click;
|
|
_connectionControl.TOOL_APMFirmware.SelectedIndexChanged += this.TOOL_APMFirmware_SelectedIndexChanged;
|
|
|
|
_connectionControl.ShowLinkStats += (sender, e) => ShowConnectionStatsForm();
|
|
srtm.datadirectory = Path.GetDirectoryName(Application.ExecutablePath) + Path.DirectorySeparatorChar + "srtm";
|
|
|
|
var t = Type.GetType("Mono.Runtime");
|
|
MONO = (t != null);
|
|
|
|
speechEngine = new Speech();
|
|
|
|
// proxy loader - dll load now instead of on config form load
|
|
new Transition(new TransitionType_EaseInEaseOut(2000));
|
|
|
|
MyRenderer.currentpressed = MenuFlightData;
|
|
|
|
MainMenu.Renderer = new MyRenderer();
|
|
|
|
foreach (object obj in Enum.GetValues(typeof(Firmwares)))
|
|
{
|
|
_connectionControl.TOOL_APMFirmware.Items.Add(obj);
|
|
}
|
|
|
|
if (_connectionControl.TOOL_APMFirmware.Items.Count > 0)
|
|
_connectionControl.TOOL_APMFirmware.SelectedIndex = 0;
|
|
|
|
this.Text = splash.Text;
|
|
|
|
comPort.BaseStream.BaudRate = 115200;
|
|
|
|
// ** Old
|
|
// CMB_serialport.Items.AddRange(SerialPort.GetPortNames());
|
|
// CMB_serialport.Items.Add("TCP");
|
|
// CMB_serialport.Items.Add("UDP");
|
|
// if (CMB_serialport.Items.Count > 0)
|
|
// {
|
|
// CMB_baudrate.SelectedIndex = 7;
|
|
// CMB_serialport.SelectedIndex = 0;
|
|
// }
|
|
// ** new
|
|
_connectionControl.CMB_serialport.Items.AddRange(ArdupilotMega.Comms.SerialPort.GetPortNames());
|
|
_connectionControl.CMB_serialport.Items.Add("TCP");
|
|
_connectionControl.CMB_serialport.Items.Add("UDP");
|
|
if (_connectionControl.CMB_serialport.Items.Count > 0)
|
|
{
|
|
_connectionControl.CMB_baudrate.SelectedIndex = 7;
|
|
_connectionControl.CMB_serialport.SelectedIndex = 0;
|
|
}
|
|
// ** Done
|
|
|
|
splash.Refresh();
|
|
Application.DoEvents();
|
|
|
|
// set this before we reset it
|
|
MainV2.config["NUM_tracklength"] = "200";
|
|
|
|
xmlconfig(false);
|
|
|
|
if (config.ContainsKey("language") && !string.IsNullOrEmpty((string)config["language"]))
|
|
changelanguage(CultureInfoEx.GetCultureInfo((string)config["language"]));
|
|
|
|
if (!MONO) // windows only
|
|
{
|
|
if (MainV2.config["showconsole"] != null && MainV2.config["showconsole"].ToString() == "True")
|
|
{
|
|
}
|
|
else
|
|
{
|
|
int win = FindWindow("ConsoleWindowClass", null);
|
|
ShowWindow(win, SW_HIDE); // hide window
|
|
}
|
|
}
|
|
|
|
ChangeUnits();
|
|
|
|
try
|
|
{
|
|
FlightData = new GCSViews.FlightData();
|
|
FlightPlanner = new GCSViews.FlightPlanner();
|
|
//Configuration = new GCSViews.ConfigurationView.Setup();
|
|
Simulation = new GCSViews.Simulation();
|
|
//Firmware = new GCSViews.Firmware();
|
|
//Terminal = new GCSViews.Terminal();
|
|
|
|
// preload
|
|
Python.CreateEngine();
|
|
}
|
|
catch (ArgumentException e)
|
|
{
|
|
//http://www.microsoft.com/en-us/download/details.aspx?id=16083
|
|
//System.ArgumentException: Font 'Arial' does not support style 'Regular'.
|
|
|
|
log.Fatal(e);
|
|
CustomMessageBox.Show(e.ToString() + "\n\n Please install this http://www.microsoft.com/en-us/download/details.aspx?id=16083");
|
|
this.Close();
|
|
}
|
|
catch (Exception e) { log.Fatal(e); CustomMessageBox.Show("A Major error has occured : " + e.ToString()); this.Close(); }
|
|
|
|
if (MainV2.config["CHK_GDIPlus"] != null)
|
|
GCSViews.FlightData.myhud.UseOpenGL = !bool.Parse(MainV2.config["CHK_GDIPlus"].ToString());
|
|
|
|
try
|
|
{
|
|
if (config["MainLocX"] != null && config["MainLocY"] != null)
|
|
{
|
|
this.StartPosition = FormStartPosition.Manual;
|
|
Point startpos = new Point(int.Parse(config["MainLocX"].ToString()), int.Parse(config["MainLocY"].ToString()));
|
|
this.Location = startpos;
|
|
}
|
|
|
|
if (config["MainMaximised"] != null)
|
|
{
|
|
this.WindowState = (FormWindowState)Enum.Parse(typeof(FormWindowState), config["MainMaximised"].ToString());
|
|
// dont allow minimised start state
|
|
if (this.WindowState == FormWindowState.Minimized)
|
|
{
|
|
this.WindowState = FormWindowState.Normal;
|
|
this.Location = new Point(100, 100);
|
|
}
|
|
}
|
|
|
|
if (config["MainHeight"] != null)
|
|
this.Height = int.Parse(config["MainHeight"].ToString());
|
|
if (config["MainWidth"] != null)
|
|
this.Width = int.Parse(config["MainWidth"].ToString());
|
|
|
|
if (config["CMB_rateattitude"] != null)
|
|
MainV2.comPort.MAV.cs.rateattitude = byte.Parse(config["CMB_rateattitude"].ToString());
|
|
if (config["CMB_rateposition"] != null)
|
|
MainV2.comPort.MAV.cs.rateposition = byte.Parse(config["CMB_rateposition"].ToString());
|
|
if (config["CMB_ratestatus"] != null)
|
|
MainV2.comPort.MAV.cs.ratestatus = byte.Parse(config["CMB_ratestatus"].ToString());
|
|
if (config["CMB_raterc"] != null)
|
|
MainV2.comPort.MAV.cs.raterc = byte.Parse(config["CMB_raterc"].ToString());
|
|
if (config["CMB_ratesensors"] != null)
|
|
MainV2.comPort.MAV.cs.ratesensors = byte.Parse(config["CMB_ratesensors"].ToString());
|
|
|
|
if (config["speechenable"] != null)
|
|
MainV2.speechEnable = bool.Parse(config["speechenable"].ToString());
|
|
|
|
//int fixme;
|
|
/*
|
|
MainV2.comPort.MAV.cs.rateattitude = 50;
|
|
MainV2.comPort.MAV.cs.rateposition = 50;
|
|
MainV2.comPort.MAV.cs.ratestatus = 50;
|
|
MainV2.comPort.MAV.cs.raterc = 50;
|
|
MainV2.comPort.MAV.cs.ratesensors = 50;
|
|
*/
|
|
try
|
|
{
|
|
if (config["TXT_homelat"] != null)
|
|
MainV2.comPort.MAV.cs.HomeLocation.Lat = double.Parse(config["TXT_homelat"].ToString());
|
|
|
|
if (config["TXT_homelng"] != null)
|
|
MainV2.comPort.MAV.cs.HomeLocation.Lng = double.Parse(config["TXT_homelng"].ToString());
|
|
|
|
if (config["TXT_homealt"] != null)
|
|
MainV2.comPort.MAV.cs.HomeLocation.Alt = double.Parse(config["TXT_homealt"].ToString());
|
|
}
|
|
catch { }
|
|
|
|
}
|
|
catch { }
|
|
|
|
if (MainV2.comPort.MAV.cs.rateattitude == 0) // initilised to 10, configured above from save
|
|
{
|
|
CustomMessageBox.Show("NOTE: your attitude rate is 0, the hud will not work\nChange in Configuration > Planner > Telemetry Rates");
|
|
}
|
|
|
|
//System.Threading.Thread.Sleep(2000);
|
|
|
|
// make sure new enough .net framework is installed
|
|
if (!MONO)
|
|
{
|
|
Microsoft.Win32.RegistryKey installed_versions = Microsoft.Win32.Registry.LocalMachine.OpenSubKey(@"SOFTWARE\Microsoft\NET Framework Setup\NDP");
|
|
string[] version_names = installed_versions.GetSubKeyNames();
|
|
//version names start with 'v', eg, 'v3.5' which needs to be trimmed off before conversion
|
|
double Framework = Convert.ToDouble(version_names[version_names.Length - 1].Remove(0, 1), CultureInfo.InvariantCulture);
|
|
int SP = Convert.ToInt32(installed_versions.OpenSubKey(version_names[version_names.Length - 1]).GetValue("SP", 0));
|
|
|
|
if (Framework < 3.5)
|
|
{
|
|
CustomMessageBox.Show("This program requires .NET Framework 3.5. You currently have " + Framework);
|
|
}
|
|
}
|
|
|
|
Application.DoEvents();
|
|
|
|
|
|
splash.Close();
|
|
}
|
|
|
|
|
|
private void ResetConnectionStats()
|
|
{
|
|
// If the form has been closed, or never shown before, we need do nothing, as
|
|
// connection stats will be reset when shown
|
|
if (this.connectionStatsForm != null && connectionStatsForm.Visible)
|
|
{
|
|
// else the form is already showing. reset the stats
|
|
this.connectionStatsForm.Controls.Clear();
|
|
_connectionStats = new ConnectionStats(comPort);
|
|
this.connectionStatsForm.Controls.Add(_connectionStats);
|
|
ThemeManager.ApplyThemeTo(this.connectionStatsForm);
|
|
}
|
|
}
|
|
|
|
private void ShowConnectionStatsForm()
|
|
{
|
|
if (this.connectionStatsForm == null || this.connectionStatsForm.IsDisposed)
|
|
{
|
|
// If the form has been closed, or never shown before, we need all new stuff
|
|
this.connectionStatsForm = new Form
|
|
{
|
|
Width = 430,
|
|
Height = 180,
|
|
MaximizeBox = false,
|
|
MinimizeBox = false,
|
|
FormBorderStyle = FormBorderStyle.FixedDialog,
|
|
Text = "Link Stats"
|
|
};
|
|
// Change the connection stats control, so that when/if the connection stats form is showing,
|
|
// there will be something to see
|
|
this.connectionStatsForm.Controls.Clear();
|
|
_connectionStats = new ConnectionStats(comPort);
|
|
this.connectionStatsForm.Controls.Add(_connectionStats);
|
|
this.connectionStatsForm.Width = _connectionStats.Width;
|
|
}
|
|
|
|
this.connectionStatsForm.Show();
|
|
ThemeManager.ApplyThemeTo(this.connectionStatsForm);
|
|
}
|
|
|
|
/// <summary>
|
|
/// used to create planner screenshots - access by control-s
|
|
/// </summary>
|
|
internal void ScreenShot()
|
|
{
|
|
Rectangle bounds = Screen.GetBounds(Point.Empty);
|
|
using (Bitmap bitmap = new Bitmap(bounds.Width, bounds.Height))
|
|
{
|
|
using (Graphics g = Graphics.FromImage(bitmap))
|
|
{
|
|
g.CopyFromScreen(Point.Empty, Point.Empty, bounds.Size);
|
|
}
|
|
string name = "ss" + DateTime.Now.ToString("HHmmss") + ".jpg";
|
|
bitmap.Save(Path.GetDirectoryName(Application.ExecutablePath) + Path.DirectorySeparatorChar + name, System.Drawing.Imaging.ImageFormat.Jpeg);
|
|
CustomMessageBox.Show("Screenshot saved to " + name);
|
|
}
|
|
|
|
}
|
|
|
|
private void CMB_serialport_Click(object sender, EventArgs e)
|
|
{
|
|
string oldport = _connectionControl.CMB_serialport.Text;
|
|
_connectionControl.CMB_serialport.Items.Clear();
|
|
_connectionControl.CMB_serialport.Items.AddRange(ArdupilotMega.Comms.SerialPort.GetPortNames());
|
|
_connectionControl.CMB_serialport.Items.Add("TCP");
|
|
_connectionControl.CMB_serialport.Items.Add("UDP");
|
|
if (_connectionControl.CMB_serialport.Items.Contains(oldport))
|
|
_connectionControl.CMB_serialport.Text = oldport;
|
|
}
|
|
|
|
|
|
private void MenuFlightData_Click(object sender, EventArgs e)
|
|
{
|
|
MyView.ShowScreen("FlightData");
|
|
}
|
|
|
|
private void MenuFlightPlanner_Click(object sender, EventArgs e)
|
|
{
|
|
// refresh ap/ac specific items
|
|
FlightPlanner.updateCMDParams();
|
|
|
|
MyView.ShowScreen("FlightPlanner");
|
|
}
|
|
|
|
private void MenuConfiguration_Click(object sender, EventArgs e)
|
|
{
|
|
MyView.ShowScreen("Config");
|
|
}
|
|
|
|
private void MenuSimulation_Click(object sender, EventArgs e)
|
|
{
|
|
MyView.ShowScreen("Simulation");
|
|
}
|
|
|
|
private void MenuFirmware_Click(object sender, EventArgs e)
|
|
{
|
|
MyView.ShowScreen("Firmware");
|
|
}
|
|
|
|
private void MenuTerminal_Click(object sender, EventArgs e)
|
|
{
|
|
if (comPort.BaseStream.IsOpen)
|
|
{
|
|
MenuConnect_Click(sender, e);
|
|
}
|
|
|
|
MyView.ShowScreen("Terminal");
|
|
}
|
|
|
|
private void MenuConnect_Click(object sender, EventArgs e)
|
|
{
|
|
giveComport = false;
|
|
|
|
// sanity check
|
|
if (comPort.BaseStream.IsOpen && MainV2.comPort.MAV.cs.groundspeed > 4)
|
|
{
|
|
if (DialogResult.No == CustomMessageBox.Show("Your model is still moving are you sure you want to disconnect?", "Disconnect", MessageBoxButtons.YesNo))
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
|
|
// cleanup from any previous sessions
|
|
if (comPort.logfile != null)
|
|
comPort.logfile.Close();
|
|
|
|
if (comPort.rawlogfile != null)
|
|
comPort.rawlogfile.Close();
|
|
|
|
comPort.logfile = null;
|
|
comPort.rawlogfile = null;
|
|
|
|
// decide if this is a connect or disconnect
|
|
if (comPort.BaseStream.IsOpen)
|
|
{
|
|
try
|
|
{
|
|
if (speechEngine != null) // cancel all pending speech
|
|
speechEngine.SpeakAsyncCancelAll();
|
|
|
|
comPort.BaseStream.DtrEnable = false;
|
|
comPort.Close();
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
log.Error(ex);
|
|
}
|
|
|
|
// now that we have closed the connection, cancel the connection stats
|
|
// so that the 'time connected' etc does not grow, but the user can still
|
|
// look at the now frozen stats on the still open form
|
|
try
|
|
{
|
|
// if terminal is used, then closed using this button.... exception
|
|
((ConnectionStats)this.connectionStatsForm.Controls[0]).StopUpdates();
|
|
}
|
|
catch { }
|
|
|
|
this.MenuConnect.BackgroundImage = global::ArdupilotMega.Properties.Resources.connect;
|
|
}
|
|
else
|
|
{
|
|
switch (_connectionControl.CMB_serialport.Text)
|
|
{
|
|
case "TCP":
|
|
comPort.BaseStream = new TcpSerial();
|
|
break;
|
|
case "UDP":
|
|
comPort.BaseStream = new UdpSerial();
|
|
break;
|
|
default:
|
|
comPort.BaseStream = new Comms.SerialPort();
|
|
break;
|
|
}
|
|
|
|
// Tell the connection UI that we are now connected.
|
|
_connectionControl.IsConnected(true);
|
|
|
|
// Here we want to reset the connection stats counter etc.
|
|
this.ResetConnectionStats();
|
|
|
|
//cleanup any log being played
|
|
comPort.logreadmode = false;
|
|
if (comPort.logplaybackfile != null)
|
|
comPort.logplaybackfile.Close();
|
|
comPort.logplaybackfile = null;
|
|
|
|
try
|
|
{
|
|
// set port, then options
|
|
comPort.BaseStream.PortName = _connectionControl.CMB_serialport.Text;
|
|
|
|
comPort.BaseStream.DataBits = 8;
|
|
comPort.BaseStream.StopBits = (StopBits)Enum.Parse(typeof(StopBits), "1");
|
|
comPort.BaseStream.Parity = (Parity)Enum.Parse(typeof(Parity), "None");
|
|
try
|
|
{
|
|
comPort.BaseStream.BaudRate = int.Parse(_connectionControl.CMB_baudrate.Text);
|
|
}
|
|
catch (Exception exp) { log.Error(exp); }
|
|
|
|
// false here
|
|
comPort.BaseStream.DtrEnable = false;
|
|
comPort.BaseStream.RtsEnable = false;
|
|
|
|
// prevent serialreader from doing anything
|
|
giveComport = true;
|
|
|
|
// reset on connect logic.
|
|
if (config["CHK_resetapmonconnect"] == null || bool.Parse(config["CHK_resetapmonconnect"].ToString()) == true)
|
|
comPort.BaseStream.toggleDTR();
|
|
|
|
giveComport = false;
|
|
|
|
// setup to record new logs
|
|
try
|
|
{
|
|
Directory.CreateDirectory(Path.GetDirectoryName(Application.ExecutablePath) + Path.DirectorySeparatorChar + @"logs");
|
|
comPort.logfile = new BinaryWriter(File.Open(Path.GetDirectoryName(Application.ExecutablePath) + Path.DirectorySeparatorChar + @"logs" + Path.DirectorySeparatorChar + DateTime.Now.ToString("yyyy-MM-dd HH-mm-ss") + ".tlog", FileMode.CreateNew, FileAccess.ReadWrite, FileShare.None));
|
|
|
|
comPort.rawlogfile = new BinaryWriter(File.Open(Path.GetDirectoryName(Application.ExecutablePath) + Path.DirectorySeparatorChar + @"logs" + Path.DirectorySeparatorChar + DateTime.Now.ToString("yyyy-MM-dd HH-mm-ss") + ".rlog", FileMode.CreateNew, FileAccess.ReadWrite, FileShare.None));
|
|
}
|
|
catch (Exception exp2) { log.Error(exp2); CustomMessageBox.Show("Failed to create log - wont log this session"); } // soft fail
|
|
|
|
// reset connect time - for timeout functions
|
|
connecttime = DateTime.Now;
|
|
|
|
// do the connect
|
|
comPort.Open(true);
|
|
|
|
// detect firmware we are conected to.
|
|
if (comPort.MAV.param["SYSID_SW_TYPE"] != null)
|
|
{
|
|
if (float.Parse(comPort.MAV.param["SYSID_SW_TYPE"].ToString()) == 10)
|
|
{
|
|
_connectionControl.TOOL_APMFirmware.SelectedIndex = _connectionControl.TOOL_APMFirmware.Items.IndexOf(Firmwares.ArduCopter2);
|
|
}
|
|
else if (float.Parse(comPort.MAV.param["SYSID_SW_TYPE"].ToString()) == 20)
|
|
{
|
|
_connectionControl.TOOL_APMFirmware.SelectedIndex = _connectionControl.TOOL_APMFirmware.Items.IndexOf(Firmwares.ArduRover);
|
|
}
|
|
else if (float.Parse(comPort.MAV.param["SYSID_SW_TYPE"].ToString()) == 0)
|
|
{
|
|
_connectionControl.TOOL_APMFirmware.SelectedIndex = _connectionControl.TOOL_APMFirmware.Items.IndexOf(Firmwares.ArduPlane);
|
|
}
|
|
}
|
|
|
|
// save the baudrate for this port
|
|
config[_connectionControl.CMB_serialport.Text + "_BAUD"] = _connectionControl.CMB_baudrate.Text;
|
|
|
|
// refresh config window if needed
|
|
if (MyView.current != null && MyView.current.Name == "Config")
|
|
{
|
|
MyView.ShowScreen("Config");
|
|
}
|
|
|
|
|
|
// load wps on connect option.
|
|
if (config["loadwpsonconnect"] != null && bool.Parse(config["loadwpsonconnect"].ToString()) == true)
|
|
{
|
|
MenuFlightPlanner_Click(null, null);
|
|
FlightPlanner.BUT_read_Click(null, null);
|
|
}
|
|
|
|
// set connected icon
|
|
this.MenuConnect.BackgroundImage = global::ArdupilotMega.Properties.Resources.disconnect;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
log.Warn(ex);
|
|
try
|
|
{
|
|
_connectionControl.IsConnected(false);
|
|
UpdateConnectIcon();
|
|
comPort.Close();
|
|
}
|
|
catch { }
|
|
// detect firmware -> scan eeprom contents -> error if no valid ap param/apvar header detected.
|
|
try
|
|
{
|
|
string version = ArduinoDetect.DetectVersion(comPort.BaseStream.PortName);
|
|
ArduinoComms port = new ArduinoSTK();
|
|
if (version == "1280")
|
|
{
|
|
port = new ArduinoSTK();
|
|
port.BaudRate = 57600;
|
|
}
|
|
else if (version == "2560")
|
|
{
|
|
port = new ArduinoSTKv2();
|
|
port.BaudRate = 115200;
|
|
}
|
|
else { throw new Exception("Can not determine APM board type"); }
|
|
port.PortName = comPort.BaseStream.PortName;
|
|
port.DtrEnable = true;
|
|
port.Open();
|
|
if (port.connectAP())
|
|
{
|
|
byte[] buffer = port.download(20);
|
|
port.Close();
|
|
|
|
if ((buffer[0] == 'A' || buffer[0] == 'P') && (buffer[1] == 'A' || buffer[1] == 'P')) // this is the apvar header
|
|
{
|
|
log.Info("Valid eeprom contents");
|
|
}
|
|
else
|
|
{
|
|
CustomMessageBox.Show("You dont appear to have uploaded a firmware yet,\n\nPlease goto the firmware page and upload one.");
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
log.Error("Could not download eeprom contents");
|
|
}
|
|
}
|
|
catch (Exception exp3) { log.Error(exp3); }
|
|
CustomMessageBox.Show("Can not establish a connection\n\n" + ex.Message);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
private void CMB_serialport_SelectedIndexChanged(object sender, EventArgs e)
|
|
{
|
|
comPortName = _connectionControl.CMB_serialport.Text;
|
|
if (comPortName == "UDP" || comPortName == "TCP")
|
|
{
|
|
_connectionControl.CMB_baudrate.Enabled = false;
|
|
if (comPortName == "TCP")
|
|
MainV2.comPort.BaseStream = new TcpSerial();
|
|
if (comPortName == "UDP")
|
|
MainV2.comPort.BaseStream = new UdpSerial();
|
|
}
|
|
else
|
|
{
|
|
_connectionControl.CMB_baudrate.Enabled = true;
|
|
MainV2.comPort.BaseStream = new ArdupilotMega.Comms.SerialPort();
|
|
}
|
|
|
|
try
|
|
{
|
|
comPort.BaseStream.PortName = _connectionControl.CMB_serialport.Text;
|
|
|
|
MainV2.comPort.BaseStream.BaudRate = int.Parse(_connectionControl.CMB_baudrate.Text);
|
|
|
|
// check for saved baud rate and restore
|
|
if (config[_connectionControl.CMB_serialport.Text + "_BAUD"] != null)
|
|
{
|
|
_connectionControl.CMB_baudrate.Text = config[_connectionControl.CMB_serialport.Text + "_BAUD"].ToString();
|
|
}
|
|
}
|
|
catch { }
|
|
}
|
|
|
|
private void MainV2_FormClosed(object sender, FormClosedEventArgs e)
|
|
{
|
|
// shutdown threads
|
|
GCSViews.FlightData.threadrun = 0;
|
|
GCSViews.Simulation.threadrun = 0;
|
|
|
|
// shutdown local thread
|
|
serialThread = false;
|
|
|
|
try
|
|
{
|
|
if (comPort.BaseStream.IsOpen)
|
|
comPort.Close();
|
|
}
|
|
catch { } // i get alot of these errors, the port is still open, but not valid - user has unpluged usb
|
|
try
|
|
{
|
|
FlightData.Dispose();
|
|
}
|
|
catch { }
|
|
try
|
|
{
|
|
FlightPlanner.Dispose();
|
|
}
|
|
catch { }
|
|
try
|
|
{
|
|
Simulation.Dispose();
|
|
}
|
|
catch { }
|
|
|
|
|
|
// save config
|
|
xmlconfig(true);
|
|
}
|
|
|
|
|
|
private void xmlconfig(bool write)
|
|
{
|
|
if (write || !File.Exists(Path.GetDirectoryName(Application.ExecutablePath) + Path.DirectorySeparatorChar + @"config.xml"))
|
|
{
|
|
try
|
|
{
|
|
XmlTextWriter xmlwriter = new XmlTextWriter(Path.GetDirectoryName(Application.ExecutablePath) + Path.DirectorySeparatorChar + @"config.xml", Encoding.ASCII);
|
|
xmlwriter.Formatting = Formatting.Indented;
|
|
|
|
xmlwriter.WriteStartDocument();
|
|
|
|
xmlwriter.WriteStartElement("Config");
|
|
|
|
xmlwriter.WriteElementString("comport", comPortName);
|
|
|
|
xmlwriter.WriteElementString("baudrate", _connectionControl.CMB_baudrate.Text);
|
|
|
|
xmlwriter.WriteElementString("APMFirmware", MainV2.comPort.MAV.cs.firmware.ToString());
|
|
|
|
foreach (string key in config.Keys)
|
|
{
|
|
try
|
|
{
|
|
if (key == "" || key.Contains("/")) // "/dev/blah"
|
|
continue;
|
|
xmlwriter.WriteElementString(key, config[key].ToString());
|
|
}
|
|
catch { }
|
|
}
|
|
|
|
xmlwriter.WriteEndElement();
|
|
|
|
xmlwriter.WriteEndDocument();
|
|
xmlwriter.Close();
|
|
}
|
|
catch (Exception ex) { CustomMessageBox.Show(ex.ToString()); }
|
|
}
|
|
else
|
|
{
|
|
try
|
|
{
|
|
using (XmlTextReader xmlreader = new XmlTextReader(Path.GetDirectoryName(Application.ExecutablePath) + Path.DirectorySeparatorChar + @"config.xml"))
|
|
{
|
|
while (xmlreader.Read())
|
|
{
|
|
xmlreader.MoveToElement();
|
|
try
|
|
{
|
|
switch (xmlreader.Name)
|
|
{
|
|
case "comport":
|
|
string temp = xmlreader.ReadString();
|
|
|
|
_connectionControl.CMB_serialport.SelectedIndex = _connectionControl.CMB_serialport.FindString(temp);
|
|
if (_connectionControl.CMB_serialport.SelectedIndex == -1)
|
|
{
|
|
_connectionControl.CMB_serialport.Text = temp; // allows ports that dont exist - yet
|
|
}
|
|
comPort.BaseStream.PortName = temp;
|
|
comPortName = temp;
|
|
break;
|
|
case "baudrate":
|
|
string temp2 = xmlreader.ReadString();
|
|
|
|
_connectionControl.CMB_baudrate.SelectedIndex = _connectionControl.CMB_baudrate.FindString(temp2);
|
|
if (_connectionControl.CMB_baudrate.SelectedIndex == -1)
|
|
{
|
|
_connectionControl.CMB_baudrate.Text = temp2;
|
|
//CMB_baudrate.SelectedIndex = CMB_baudrate.FindString("57600"); ; // must exist
|
|
}
|
|
//bau = int.Parse(CMB_baudrate.Text);
|
|
break;
|
|
case "APMFirmware":
|
|
string temp3 = xmlreader.ReadString();
|
|
_connectionControl.TOOL_APMFirmware.SelectedIndex = _connectionControl.TOOL_APMFirmware.FindStringExact(temp3);
|
|
if (_connectionControl.TOOL_APMFirmware.SelectedIndex == -1)
|
|
_connectionControl.TOOL_APMFirmware.SelectedIndex = 0;
|
|
MainV2.comPort.MAV.cs.firmware = (MainV2.Firmwares)Enum.Parse(typeof(MainV2.Firmwares), _connectionControl.TOOL_APMFirmware.Text);
|
|
break;
|
|
case "Config":
|
|
break;
|
|
case "xml":
|
|
break;
|
|
default:
|
|
if (xmlreader.Name == "") // line feeds
|
|
break;
|
|
config[xmlreader.Name] = xmlreader.ReadString();
|
|
break;
|
|
}
|
|
}
|
|
// silent fail on bad entry
|
|
catch (Exception ee)
|
|
{
|
|
log.Error(ee);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
log.Error("Bad Config File", ex);
|
|
}
|
|
}
|
|
}
|
|
|
|
private void CMB_baudrate_SelectedIndexChanged(object sender, EventArgs e)
|
|
{
|
|
try
|
|
{
|
|
comPort.BaseStream.BaudRate = int.Parse(_connectionControl.CMB_baudrate.Text);
|
|
}
|
|
catch
|
|
{
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// thread used to send joystick packets to the MAV
|
|
/// </summary>
|
|
private void joysticksend()
|
|
{
|
|
|
|
float rate = 50;
|
|
int count = 0;
|
|
|
|
DateTime lastratechange = DateTime.Now;
|
|
|
|
while (true)
|
|
{
|
|
try
|
|
{
|
|
if (MONO)
|
|
{
|
|
log.Error("Mono: closing joystick thread");
|
|
break;
|
|
}
|
|
|
|
if (!MONO)
|
|
{
|
|
//joystick stuff
|
|
|
|
if (joystick != null && joystick.enabled)
|
|
{
|
|
MAVLink.mavlink_rc_channels_override_t rc = new MAVLink.mavlink_rc_channels_override_t();
|
|
|
|
rc.target_component = comPort.MAV.compid;
|
|
rc.target_system = comPort.MAV.sysid;
|
|
|
|
if (joystick.getJoystickAxis(1) != Joystick.joystickaxis.None)
|
|
rc.chan1_raw = MainV2.comPort.MAV.cs.rcoverridech1;//(ushort)(((int)state.Rz / 65.535) + 1000);
|
|
if (joystick.getJoystickAxis(2) != Joystick.joystickaxis.None)
|
|
rc.chan2_raw = MainV2.comPort.MAV.cs.rcoverridech2;//(ushort)(((int)state.Y / 65.535) + 1000);
|
|
if (joystick.getJoystickAxis(3) != Joystick.joystickaxis.None)
|
|
rc.chan3_raw = MainV2.comPort.MAV.cs.rcoverridech3;//(ushort)(1000 - ((int)slider[0] / 65.535 ) + 1000);
|
|
if (joystick.getJoystickAxis(4) != Joystick.joystickaxis.None)
|
|
rc.chan4_raw = MainV2.comPort.MAV.cs.rcoverridech4;//(ushort)(((int)state.X / 65.535) + 1000);
|
|
if (joystick.getJoystickAxis(5) != Joystick.joystickaxis.None)
|
|
rc.chan5_raw = MainV2.comPort.MAV.cs.rcoverridech5;
|
|
if (joystick.getJoystickAxis(6) != Joystick.joystickaxis.None)
|
|
rc.chan6_raw = MainV2.comPort.MAV.cs.rcoverridech6;
|
|
if (joystick.getJoystickAxis(7) != Joystick.joystickaxis.None)
|
|
rc.chan7_raw = MainV2.comPort.MAV.cs.rcoverridech7;
|
|
if (joystick.getJoystickAxis(8) != Joystick.joystickaxis.None)
|
|
rc.chan8_raw = MainV2.comPort.MAV.cs.rcoverridech8;
|
|
|
|
if (lastjoystick.AddMilliseconds(rate) < DateTime.Now)
|
|
{
|
|
/*
|
|
if (MainV2.comPort.MAV.cs.rssi > 0 && MainV2.comPort.MAV.cs.remrssi > 0)
|
|
{
|
|
if (lastratechange.Second != DateTime.Now.Second)
|
|
{
|
|
if (MainV2.comPort.MAV.cs.txbuffer > 90)
|
|
{
|
|
if (rate < 20)
|
|
rate = 21;
|
|
rate--;
|
|
|
|
if (MainV2.comPort.MAV.cs.linkqualitygcs < 70)
|
|
rate = 50;
|
|
}
|
|
else
|
|
{
|
|
if (rate > 100)
|
|
rate = 100;
|
|
rate++;
|
|
}
|
|
|
|
lastratechange = DateTime.Now;
|
|
}
|
|
|
|
}
|
|
*/
|
|
// Console.WriteLine(DateTime.Now.Millisecond + " {0} {1} {2} {3} {4}", rc.chan1_raw, rc.chan2_raw, rc.chan3_raw, rc.chan4_raw,rate);
|
|
comPort.sendPacket(rc);
|
|
count++;
|
|
lastjoystick = DateTime.Now;
|
|
}
|
|
|
|
}
|
|
}
|
|
Thread.Sleep(20);
|
|
}
|
|
catch
|
|
{
|
|
|
|
} // cant fall out
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Used to fix the icon status for unexpected unplugs etc...
|
|
/// </summary>
|
|
private void UpdateConnectIcon()
|
|
{
|
|
if ((DateTime.Now - connectButtonUpdate).Milliseconds > 500)
|
|
{
|
|
// Console.WriteLine(DateTime.Now.Millisecond);
|
|
if (comPort.BaseStream.IsOpen)
|
|
{
|
|
if ((string)this.MenuConnect.BackgroundImage.Tag != "Disconnect")
|
|
{
|
|
this.BeginInvoke((MethodInvoker)delegate
|
|
{
|
|
this.MenuConnect.BackgroundImage = global::ArdupilotMega.Properties.Resources.disconnect;
|
|
this.MenuConnect.BackgroundImage.Tag = "Disconnect";
|
|
_connectionControl.IsConnected(true);
|
|
});
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ((string)this.MenuConnect.BackgroundImage.Tag != "Connect")
|
|
{
|
|
this.BeginInvoke((MethodInvoker)delegate
|
|
{
|
|
this.MenuConnect.BackgroundImage = global::ArdupilotMega.Properties.Resources.connect;
|
|
this.MenuConnect.BackgroundImage.Tag = "Connect";
|
|
_connectionControl.IsConnected(false);
|
|
if (_connectionStats != null)
|
|
{
|
|
_connectionStats.StopUpdates();
|
|
}
|
|
});
|
|
}
|
|
|
|
if (comPort.logreadmode)
|
|
{
|
|
this.BeginInvoke((MethodInvoker)delegate
|
|
{
|
|
_connectionControl.IsConnected(true);
|
|
});
|
|
}
|
|
}
|
|
connectButtonUpdate = DateTime.Now;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// main serial reader thread
|
|
/// controls
|
|
/// serial reading
|
|
/// link quality stats
|
|
/// speech voltage - custom - alt warning - data lost
|
|
/// heartbeat packet sending
|
|
///
|
|
/// and can't fall out
|
|
/// </summary>
|
|
private void SerialReader()
|
|
{
|
|
if (serialThread == true)
|
|
return;
|
|
serialThread = true;
|
|
|
|
int minbytes = 0;
|
|
|
|
DateTime speechcustomtime = DateTime.Now;
|
|
|
|
DateTime linkqualitytime = DateTime.Now;
|
|
|
|
while (serialThread)
|
|
{
|
|
try
|
|
{
|
|
Thread.Sleep(5);
|
|
|
|
// update connect/disconnect button and info stats
|
|
UpdateConnectIcon();
|
|
|
|
// 30 seconds interval speech options
|
|
if (speechEnable && speechEngine != null && (DateTime.Now - speechcustomtime).TotalSeconds > 30 && MainV2.comPort.MAV.cs.lat != 0 && (MainV2.comPort.logreadmode || comPort.BaseStream.IsOpen))
|
|
{
|
|
//speechbatteryvolt
|
|
float warnvolt = 0;
|
|
float.TryParse(MainV2.getConfig("speechbatteryvolt"), out warnvolt);
|
|
|
|
if (MainV2.getConfig("speechbatteryenabled") == "True" && MainV2.comPort.MAV.cs.battery_voltage <= warnvolt)
|
|
{
|
|
MainV2.speechEngine.SpeakAsync(Common.speechConversion(MainV2.getConfig("speechbattery")));
|
|
}
|
|
|
|
if (MainV2.getConfig("speechcustomenabled") == "True")
|
|
{
|
|
MainV2.speechEngine.SpeakAsync(Common.speechConversion(MainV2.getConfig("speechcustom")));
|
|
}
|
|
|
|
speechcustomtime = DateTime.Now;
|
|
}
|
|
|
|
// speech altitude warning.
|
|
if (speechEnable && speechEngine != null && (MainV2.comPort.logreadmode || comPort.BaseStream.IsOpen))
|
|
{
|
|
float warnalt = float.MaxValue;
|
|
float.TryParse(MainV2.getConfig("speechaltheight"), out warnalt);
|
|
try
|
|
{
|
|
if (MainV2.getConfig("speechaltenabled") == "True" && (MainV2.comPort.MAV.cs.alt - (int)double.Parse(MainV2.getConfig("TXT_homealt"))) <= warnalt)
|
|
{
|
|
if (MainV2.speechEngine.State == SynthesizerState.Ready)
|
|
MainV2.speechEngine.SpeakAsync(Common.speechConversion(MainV2.getConfig("speechalt")));
|
|
}
|
|
}
|
|
catch { } // silent fail
|
|
}
|
|
|
|
// if not connected or busy, sleep and loop
|
|
if (!comPort.BaseStream.IsOpen || giveComport == true)
|
|
{
|
|
System.Threading.Thread.Sleep(100);
|
|
continue;
|
|
}
|
|
|
|
// make sure we attenuate the link quality if we dont see any valid packets
|
|
if ((DateTime.Now - comPort.lastvalidpacket).TotalSeconds > 10)
|
|
{
|
|
MainV2.comPort.MAV.cs.linkqualitygcs = 0;
|
|
}
|
|
|
|
// attenuate the link qualty over time
|
|
if ((DateTime.Now - comPort.lastvalidpacket).TotalSeconds >= 1)
|
|
{
|
|
if (linkqualitytime.Second != DateTime.Now.Second)
|
|
{
|
|
MainV2.comPort.MAV.cs.linkqualitygcs = (ushort)(MainV2.comPort.MAV.cs.linkqualitygcs * 0.8f);
|
|
linkqualitytime = DateTime.Now;
|
|
|
|
GCSViews.FlightData.myhud.Invalidate();
|
|
}
|
|
}
|
|
|
|
// send a hb every seconds from gcs to ap
|
|
if (heatbeatSend.Second != DateTime.Now.Second)
|
|
{
|
|
MAVLink.mavlink_heartbeat_t htb = new MAVLink.mavlink_heartbeat_t()
|
|
{
|
|
type = (byte)MAVLink.MAV_TYPE.GCS,
|
|
autopilot = (byte)MAVLink.MAV_AUTOPILOT.ARDUPILOTMEGA,
|
|
mavlink_version = 3,
|
|
};
|
|
|
|
comPort.sendPacket(htb);
|
|
heatbeatSend = DateTime.Now;
|
|
}
|
|
|
|
// data loss warning - ignore first 30 seconds of connect
|
|
if ((DateTime.Now - comPort.lastvalidpacket).TotalSeconds > 10 && (DateTime.Now - connecttime).TotalSeconds > 30)
|
|
{
|
|
if (speechEnable && speechEngine != null)
|
|
{
|
|
if (MainV2.speechEngine.State == SynthesizerState.Ready)
|
|
MainV2.speechEngine.SpeakAsync("WARNING No Data for " + (int)(DateTime.Now - comPort.lastvalidpacket).TotalSeconds + " Seconds");
|
|
}
|
|
}
|
|
|
|
// actauly read the packets
|
|
while (comPort.BaseStream.BytesToRead > minbytes && giveComport == false)
|
|
{
|
|
try
|
|
{
|
|
comPort.readPacket();
|
|
}
|
|
catch { }
|
|
}
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
log.Error("Serial Reader fail :" + e.ToString());
|
|
try
|
|
{
|
|
comPort.Close();
|
|
}
|
|
catch { }
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Override the stock ToolStripProfessionalRenderer to implement 'highlighting' of the
|
|
/// currently selected GCS view.
|
|
/// </summary>
|
|
private class MyRenderer : ToolStripProfessionalRenderer
|
|
{
|
|
public static ToolStripItem currentpressed;
|
|
protected override void OnRenderButtonBackground(ToolStripItemRenderEventArgs e)
|
|
{
|
|
//BackgroundImage
|
|
if (e.Item.BackgroundImage == null) base.OnRenderButtonBackground(e);
|
|
else
|
|
{
|
|
Rectangle bounds = new Rectangle(Point.Empty, e.Item.Size);
|
|
e.Graphics.DrawImage(e.Item.BackgroundImage, bounds);
|
|
if (e.Item.Pressed || e.Item == currentpressed)
|
|
{
|
|
SolidBrush brush = new SolidBrush(Color.FromArgb(73, 0x2b, 0x3a, 0x03));
|
|
e.Graphics.FillRectangle(brush, bounds);
|
|
if (e.Item.Name != "MenuConnect")
|
|
{
|
|
//Console.WriteLine("new " + e.Item.Name + " old " + currentpressed.Name );
|
|
//e.Item.GetCurrentParent().Invalidate();
|
|
if (currentpressed != e.Item)
|
|
currentpressed.Invalidate();
|
|
currentpressed = e.Item;
|
|
}
|
|
|
|
// Something...
|
|
}
|
|
else if (e.Item.Selected) // mouse over
|
|
{
|
|
SolidBrush brush = new SolidBrush(Color.FromArgb(73, 0x2b, 0x3a, 0x03));
|
|
e.Graphics.FillRectangle(brush, bounds);
|
|
// Something...
|
|
}
|
|
using (Pen pen = new Pen(Color.Black))
|
|
{
|
|
//e.GraphiMainV2.comPort.MAV.cs.DrawRectangle(pen, bounds.X, bounds.Y, bounds.Width - 1, bounds.Height - 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
protected override void OnRenderItemImage(ToolStripItemImageRenderEventArgs e)
|
|
{
|
|
//base.OnRenderItemImage(e);
|
|
}
|
|
}
|
|
|
|
private void MainV2_Load(object sender, EventArgs e)
|
|
{
|
|
// check if its defined, and force to show it if not known about
|
|
if (config["menu_autohide"] == null)
|
|
{
|
|
config["menu_autohide"] = "false";
|
|
}
|
|
|
|
try
|
|
{
|
|
AutoHideMenu(bool.Parse(config["menu_autohide"].ToString()));
|
|
}
|
|
catch { }
|
|
|
|
MyView.AddScreen(new MainSwitcher.Screen("FlightData", FlightData, true));
|
|
MyView.AddScreen(new MainSwitcher.Screen("FlightPlanner", FlightPlanner, true));
|
|
MyView.AddScreen(new MainSwitcher.Screen("Config", new GCSViews.ConfigurationView.Setup(), false));
|
|
MyView.AddScreen(new MainSwitcher.Screen("Simulation", Simulation, true));
|
|
MyView.AddScreen(new MainSwitcher.Screen("Firmware", new GCSViews.Firmware(), false));
|
|
MyView.AddScreen(new MainSwitcher.Screen("Terminal", new GCSViews.Terminal(), false));
|
|
MyView.AddScreen(new MainSwitcher.Screen("Help", new GCSViews.Help(), false));
|
|
|
|
// init button depressed - ensures correct action
|
|
//int fixme;
|
|
MenuFlightData_Click(sender, e);
|
|
|
|
|
|
// for long running tasks using own threads.
|
|
// for short use threadpool
|
|
|
|
// setup http server
|
|
try
|
|
{
|
|
listener = new TcpListener(IPAddress.Any, 56781);
|
|
new Thread(listernforclients)
|
|
{
|
|
Name = "motion jpg stream-network kml",
|
|
IsBackground = true
|
|
}.Start();
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
log.Error("Error starting TCP listener thread: ", ex);
|
|
CustomMessageBox.Show(ex.ToString());
|
|
}
|
|
|
|
/// setup joystick packet sender
|
|
new Thread(new ThreadStart(joysticksend))
|
|
{
|
|
IsBackground = true,
|
|
Priority = ThreadPriority.AboveNormal,
|
|
Name = "Main joystick sender"
|
|
}.Start();
|
|
|
|
// setup main serial reader
|
|
new Thread(SerialReader)
|
|
{
|
|
IsBackground = true,
|
|
Name = "Main Serial reader",
|
|
Priority = ThreadPriority.AboveNormal
|
|
}.Start();
|
|
|
|
try
|
|
{
|
|
if (!System.Diagnostics.Debugger.IsAttached)
|
|
{
|
|
// single update check per day
|
|
if (getConfig("update_check") != DateTime.Now.ToShortDateString())
|
|
{
|
|
CheckForUpdate();
|
|
config["update_check"] = DateTime.Now.ToShortDateString();
|
|
}
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
log.Error("Update check failed", ex);
|
|
}
|
|
}
|
|
|
|
public static String ComputeWebSocketHandshakeSecurityHash09(String secWebSocketKey)
|
|
{
|
|
const String MagicKEY = "258EAFA5-E914-47DA-95CA-C5AB0DC85B11";
|
|
String secWebSocketAccept = String.Empty;
|
|
|
|
// 1. Combine the request Sec-WebSocket-Key with magic key.
|
|
String ret = secWebSocketKey + MagicKEY;
|
|
|
|
// 2. Compute the SHA1 hash
|
|
System.Security.Cryptography.SHA1 sha = new System.Security.Cryptography.SHA1CryptoServiceProvider();
|
|
byte[] sha1Hash = sha.ComputeHash(Encoding.UTF8.GetBytes(ret));
|
|
|
|
// 3. Base64 encode the hash
|
|
secWebSocketAccept = Convert.ToBase64String(sha1Hash);
|
|
|
|
return secWebSocketAccept;
|
|
}
|
|
|
|
/// <summary>
|
|
/// little web server for sending network link kml's
|
|
/// </summary>
|
|
void listernforclients()
|
|
{
|
|
try
|
|
{
|
|
listener.Start();
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
log.Error("Exception starting lister. Possible multiple instances of planner?", e);
|
|
return;
|
|
} // in use
|
|
// Enter the listening loop.
|
|
while (true)
|
|
{
|
|
// Perform a blocking call to accept requests.
|
|
// You could also user server.AcceptSocket() here.
|
|
try
|
|
{
|
|
log.Info("Listening for client - 1 client at a time");
|
|
TcpClient client = listener.AcceptTcpClient();
|
|
// Get a stream object for reading and writing
|
|
log.Info("Accepted Client " + client.Client.RemoteEndPoint.ToString());
|
|
//client.SendBufferSize = 100 * 1024; // 100kb
|
|
//client.LingerState.Enabled = true;
|
|
//client.NoDelay = true;
|
|
|
|
// makesure we have valid image
|
|
GCSViews.FlightData.mymap.streamjpgenable = true;
|
|
GCSViews.FlightData.myhud.streamjpgenable = true;
|
|
|
|
MethodInvoker m = delegate()
|
|
{
|
|
GCSViews.FlightData.mymap.Refresh();
|
|
};
|
|
this.Invoke(m);
|
|
|
|
NetworkStream stream = client.GetStream();
|
|
|
|
var asciiEncoding = new ASCIIEncoding();
|
|
|
|
var request = new byte[1024];
|
|
|
|
int len = stream.Read(request, 0, request.Length);
|
|
string head = System.Text.Encoding.ASCII.GetString(request, 0, len);
|
|
log.Info(head);
|
|
|
|
int index = head.IndexOf('\n');
|
|
|
|
string url = head.Substring(0, index - 1);
|
|
//url = url.Replace("\r", "");
|
|
//url = url.Replace("GET ","");
|
|
//url = url.Replace(" HTTP/1.0", "");
|
|
//url = url.Replace(" HTTP/1.1", "");
|
|
|
|
if (url.Contains("websocket"))
|
|
{
|
|
using (var writer = new StreamWriter(stream, Encoding.Default))
|
|
{
|
|
writer.WriteLine("HTTP/1.1 101 WebSocket Protocol Handshake");
|
|
writer.WriteLine("Upgrade: WebSocket");
|
|
writer.WriteLine("Connection: Upgrade");
|
|
writer.WriteLine("WebSocket-Location: ws://localhost:56781/websocket/server");
|
|
|
|
int start = head.IndexOf("Sec-WebSocket-Key:") + 19;
|
|
int end = head.IndexOf('\r', start);
|
|
if (end == -1)
|
|
end = head.IndexOf('\n', start);
|
|
string accept = ComputeWebSocketHandshakeSecurityHash09(head.Substring(start, end - start));
|
|
|
|
writer.WriteLine("Sec-WebSocket-Accept: " + accept);
|
|
|
|
writer.WriteLine("Server: APM Planner");
|
|
|
|
writer.WriteLine("");
|
|
|
|
writer.Flush();
|
|
|
|
while (client.Connected)
|
|
{
|
|
Thread.Sleep(200);
|
|
log.Debug(stream.DataAvailable + " " + client.Available);
|
|
|
|
while (client.Available > 0)
|
|
{
|
|
Console.Write(stream.ReadByte());
|
|
}
|
|
|
|
byte[] packet = new byte[256];
|
|
|
|
string sendme = MainV2.comPort.MAV.cs.roll + "," + MainV2.comPort.MAV.cs.pitch + "," + MainV2.comPort.MAV.cs.yaw;
|
|
|
|
packet[0] = 0x81; // fin - binary
|
|
packet[1] = (byte)sendme.Length;
|
|
|
|
int i = 2;
|
|
foreach (char ch in sendme)
|
|
{
|
|
packet[i++] = (byte)ch;
|
|
}
|
|
|
|
stream.Write(packet, 0, i);
|
|
|
|
//break;
|
|
}
|
|
}
|
|
}
|
|
else if (url.Contains("georefnetwork.kml"))
|
|
{
|
|
string header = "HTTP/1.1 200 OK\r\nContent-Type: application/vnd.google-earth.kml+xml\r\n\r\n";
|
|
byte[] temp = asciiEncoding.GetBytes(header);
|
|
stream.Write(temp, 0, temp.Length);
|
|
|
|
byte[] buffer = Encoding.ASCII.GetBytes(georefkml);
|
|
|
|
stream.Write(buffer, 0, buffer.Length);
|
|
|
|
stream.Close();
|
|
}
|
|
else if (url.Contains("network.kml"))
|
|
{
|
|
string header = "HTTP/1.1 200 OK\r\nContent-Type: application/vnd.google-earth.kml+xml\r\n\r\n";
|
|
byte[] temp = asciiEncoding.GetBytes(header);
|
|
stream.Write(temp, 0, temp.Length);
|
|
|
|
SharpKml.Dom.Document kml = new SharpKml.Dom.Document();
|
|
|
|
SharpKml.Dom.Placemark pmplane = new SharpKml.Dom.Placemark();
|
|
pmplane.Name = "P/Q ";
|
|
|
|
pmplane.Visibility = true;
|
|
|
|
SharpKml.Dom.Location loc = new SharpKml.Dom.Location();
|
|
loc.Latitude = MainV2.comPort.MAV.cs.lat;
|
|
loc.Longitude = MainV2.comPort.MAV.cs.lng;
|
|
loc.Altitude = MainV2.comPort.MAV.cs.alt;
|
|
|
|
if (loc.Altitude < 0)
|
|
loc.Altitude = 0.01;
|
|
|
|
SharpKml.Dom.Orientation ori = new SharpKml.Dom.Orientation();
|
|
ori.Heading = MainV2.comPort.MAV.cs.yaw;
|
|
ori.Roll = -MainV2.comPort.MAV.cs.roll;
|
|
ori.Tilt = -MainV2.comPort.MAV.cs.pitch;
|
|
|
|
SharpKml.Dom.Scale sca = new SharpKml.Dom.Scale();
|
|
|
|
sca.X = 2;
|
|
sca.Y = 2;
|
|
sca.Z = 2;
|
|
|
|
SharpKml.Dom.Model model = new SharpKml.Dom.Model();
|
|
model.Location = loc;
|
|
model.Orientation = ori;
|
|
model.AltitudeMode = SharpKml.Dom.AltitudeMode.Absolute;
|
|
model.Scale = sca;
|
|
|
|
SharpKml.Dom.Link link = new SharpKml.Dom.Link();
|
|
link.Href = new Uri("block_plane_0.dae", UriKind.Relative);
|
|
|
|
model.Link = link;
|
|
|
|
pmplane.Geometry = model;
|
|
|
|
SharpKml.Dom.LookAt la = new SharpKml.Dom.LookAt()
|
|
{
|
|
Altitude = loc.Altitude.Value,
|
|
Latitude = loc.Latitude.Value,
|
|
Longitude = loc.Longitude.Value,
|
|
Tilt = 80,
|
|
Heading = MainV2.comPort.MAV.cs.yaw,
|
|
AltitudeMode = SharpKml.Dom.AltitudeMode.Absolute,
|
|
Range = 50
|
|
};
|
|
|
|
kml.Viewpoint = la;
|
|
|
|
kml.AddFeature(pmplane);
|
|
|
|
SharpKml.Dom.CoordinateCollection coords = new SharpKml.Dom.CoordinateCollection();
|
|
|
|
foreach (var point in MainV2.comPort.MAV.wps.Values)
|
|
{
|
|
coords.Add(new SharpKml.Base.Vector(point.x, point.y, point.z));
|
|
}
|
|
|
|
SharpKml.Dom.LineString ls = new SharpKml.Dom.LineString();
|
|
ls.AltitudeMode = SharpKml.Dom.AltitudeMode.RelativeToGround;
|
|
ls.Coordinates = coords;
|
|
|
|
SharpKml.Dom.Placemark pm = new SharpKml.Dom.Placemark() { Geometry = ls };
|
|
|
|
kml.AddFeature(pm);
|
|
|
|
SharpKml.Base.Serializer serializer = new SharpKml.Base.Serializer();
|
|
serializer.Serialize(kml);
|
|
|
|
byte[] buffer = Encoding.ASCII.GetBytes(serializer.Xml);
|
|
|
|
stream.Write(buffer, 0, buffer.Length);
|
|
|
|
stream.Close();
|
|
}
|
|
else if (url.Contains("block_plane_0.dae"))
|
|
{
|
|
string header = "HTTP/1.1 200 OK\r\nContent-Type: text/plain\r\n\r\n";
|
|
byte[] temp = asciiEncoding.GetBytes(header);
|
|
stream.Write(temp, 0, temp.Length);
|
|
|
|
BinaryReader file = new BinaryReader(File.Open("block_plane_0.dae", FileMode.Open, FileAccess.Read, FileShare.Read));
|
|
byte[] buffer = new byte[1024];
|
|
while (file.PeekChar() != -1)
|
|
{
|
|
|
|
int leng = file.Read(buffer, 0, buffer.Length);
|
|
|
|
stream.Write(buffer, 0, leng);
|
|
}
|
|
file.Close();
|
|
stream.Close();
|
|
}
|
|
else if (url.Contains("hud.html"))
|
|
{
|
|
string header = "HTTP/1.1 200 OK\r\nContent-Type: text/html\r\n\r\n";
|
|
byte[] temp = asciiEncoding.GetBytes(header);
|
|
stream.Write(temp, 0, temp.Length);
|
|
|
|
BinaryReader file = new BinaryReader(File.Open("hud.html", FileMode.Open, FileAccess.Read, FileShare.Read));
|
|
byte[] buffer = new byte[1024];
|
|
while (file.PeekChar() != -1)
|
|
{
|
|
|
|
int leng = file.Read(buffer, 0, buffer.Length);
|
|
|
|
stream.Write(buffer, 0, leng);
|
|
}
|
|
file.Close();
|
|
stream.Close();
|
|
}
|
|
else if (url.ToLower().Contains("hud.jpg") || url.ToLower().Contains("map.jpg") || url.ToLower().Contains("both.jpg"))
|
|
{
|
|
string header = "HTTP/1.1 200 OK\r\nContent-Type: multipart/x-mixed-replace;boundary=APMPLANNER\r\n\r\n--APMPLANNER\r\n";
|
|
byte[] temp = asciiEncoding.GetBytes(header);
|
|
stream.Write(temp, 0, temp.Length);
|
|
|
|
while (client.Connected)
|
|
{
|
|
System.Threading.Thread.Sleep(200); // 5hz
|
|
byte[] data = null;
|
|
|
|
if (url.ToLower().Contains("hud"))
|
|
{
|
|
GCSViews.FlightData.myhud.streamjpgenable = true;
|
|
data = GCSViews.FlightData.myhud.streamjpg.ToArray();
|
|
}
|
|
else if (url.ToLower().Contains("map"))
|
|
{
|
|
GCSViews.FlightData.mymap.streamjpgenable = true;
|
|
data = GCSViews.FlightData.mymap.streamjpg.ToArray();
|
|
}
|
|
else
|
|
{
|
|
GCSViews.FlightData.mymap.streamjpgenable = true;
|
|
GCSViews.FlightData.myhud.streamjpgenable = true;
|
|
Image img1 = Image.FromStream(GCSViews.FlightData.myhud.streamjpg);
|
|
Image img2 = Image.FromStream(GCSViews.FlightData.mymap.streamjpg);
|
|
int bigger = img1.Height > img2.Height ? img1.Height : img2.Height;
|
|
Image imgout = new Bitmap(img1.Width + img2.Width, bigger);
|
|
|
|
Graphics grap = Graphics.FromImage(imgout);
|
|
|
|
grap.DrawImageUnscaled(img1, 0, 0);
|
|
grap.DrawImageUnscaled(img2, img1.Width, 0);
|
|
|
|
MemoryStream streamjpg = new MemoryStream();
|
|
imgout.Save(streamjpg, System.Drawing.Imaging.ImageFormat.Jpeg);
|
|
data = streamjpg.ToArray();
|
|
|
|
}
|
|
|
|
header = "Content-Type: image/jpeg\r\nContent-Length: " + data.Length + "\r\n\r\n";
|
|
temp = asciiEncoding.GetBytes(header);
|
|
stream.Write(temp, 0, temp.Length);
|
|
|
|
stream.Write(data, 0, data.Length);
|
|
|
|
header = "\r\n--APMPLANNER\r\n";
|
|
temp = asciiEncoding.GetBytes(header);
|
|
stream.Write(temp, 0, temp.Length);
|
|
|
|
}
|
|
GCSViews.FlightData.mymap.streamjpgenable = false;
|
|
GCSViews.FlightData.myhud.streamjpgenable = false;
|
|
stream.Close();
|
|
|
|
}
|
|
else if (url.Contains("guided?"))
|
|
{
|
|
//http://127.0.0.1:56781/guided?lat=-34&lng=117.8&alt=30
|
|
|
|
Regex rex = new Regex(@"lat=([\-\.0-9]+)&lng=([\-\.0-9]+)&alt=([\.0-9]+)",RegexOptions.IgnoreCase);
|
|
|
|
Match match = rex.Match(url);
|
|
|
|
if (match.Success)
|
|
{
|
|
Locationwp gwp = new Locationwp()
|
|
{
|
|
lat = float.Parse(match.Groups[1].Value),
|
|
lng = float.Parse(match.Groups[2].Value),
|
|
alt = float.Parse(match.Groups[3].Value)
|
|
};
|
|
try
|
|
{
|
|
comPort.setGuidedModeWP(gwp);
|
|
}
|
|
catch { }
|
|
|
|
string header = "HTTP/1.1 200 OK\r\n\r\nSent Guide Mode Wp";
|
|
byte[] temp = asciiEncoding.GetBytes(header);
|
|
stream.Write(temp, 0, temp.Length);
|
|
}
|
|
else
|
|
{
|
|
string header = "HTTP/1.1 200 OK\r\n\r\nFailed Guide Mode Wp";
|
|
byte[] temp = asciiEncoding.GetBytes(header);
|
|
stream.Write(temp, 0, temp.Length);
|
|
}
|
|
|
|
}
|
|
stream.Close();
|
|
}
|
|
catch (Exception ee)
|
|
{
|
|
log.Error("Failed mjpg ", ee);
|
|
}
|
|
}
|
|
}
|
|
|
|
private void TOOL_APMFirmware_SelectedIndexChanged(object sender, EventArgs e)
|
|
{
|
|
MainV2.comPort.MAV.cs.firmware = (MainV2.Firmwares)Enum.Parse(typeof(MainV2.Firmwares), _connectionControl.TOOL_APMFirmware.Text);
|
|
}
|
|
|
|
private void MainV2_Resize(object sender, EventArgs e)
|
|
{
|
|
log.Info("myview width " + MyView.Width + " height " + MyView.Height);
|
|
log.Info("this width " + this.Width + " height " + this.Height);
|
|
}
|
|
|
|
private void MenuHelp_Click(object sender, EventArgs e)
|
|
{
|
|
MyView.ShowScreen("Help");
|
|
}
|
|
|
|
|
|
public static void updateCheckMain(ProgressReporterDialogue frmProgressReporter)
|
|
{
|
|
var baseurl = ConfigurationManager.AppSettings["UpdateLocation"];
|
|
try
|
|
{
|
|
bool update = updateCheck(frmProgressReporter, baseurl, "");
|
|
var process = new Process();
|
|
string exePath = Path.GetDirectoryName(Application.ExecutablePath);
|
|
if (MONO)
|
|
{
|
|
process.StartInfo.FileName = "mono";
|
|
process.StartInfo.Arguments = " \"" + exePath + Path.DirectorySeparatorChar + "Updater.exe\"" + " \"" + Application.ExecutablePath + "\"";
|
|
}
|
|
else
|
|
{
|
|
process.StartInfo.FileName = exePath + Path.DirectorySeparatorChar + "Updater.exe";
|
|
process.StartInfo.Arguments = Application.ExecutablePath;
|
|
}
|
|
|
|
try
|
|
{
|
|
foreach (string newupdater in Directory.GetFiles(exePath, "Updater.exe*.new"))
|
|
{
|
|
File.Copy(newupdater, newupdater.Remove(newupdater.Length - 4), true);
|
|
File.Delete(newupdater);
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
log.Error("Exception during update", ex);
|
|
}
|
|
if (frmProgressReporter != null)
|
|
frmProgressReporter.UpdateProgressAndStatus(-1, "Starting Updater");
|
|
log.Info("Starting new process: " + process.StartInfo.FileName + " with " + process.StartInfo.Arguments);
|
|
process.Start();
|
|
log.Info("Quitting existing process");
|
|
try
|
|
{
|
|
// clean close
|
|
MainV2.instance.BeginInvoke((MethodInvoker)delegate()
|
|
{
|
|
MainV2.instance.Close();
|
|
});
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
log.Error(ex);
|
|
Application.Exit();
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
log.Error("Update Failed", ex);
|
|
CustomMessageBox.Show("Update Failed " + ex.Message);
|
|
}
|
|
}
|
|
|
|
private static void UpdateLabel(Label loadinglabel, string text)
|
|
{
|
|
MainV2.instance.Invoke((MethodInvoker)delegate
|
|
{
|
|
loadinglabel.Text = text;
|
|
|
|
Application.DoEvents();
|
|
});
|
|
}
|
|
|
|
private static void CheckForUpdate()
|
|
{
|
|
var baseurl = ConfigurationManager.AppSettings["UpdateLocation"];
|
|
string path = Path.GetDirectoryName(Application.ExecutablePath);
|
|
|
|
path = path + Path.DirectorySeparatorChar + "version.txt";
|
|
|
|
log.Debug(path);
|
|
|
|
// Create a request using a URL that can receive a post.
|
|
string requestUriString = baseurl + Path.GetFileName(path);
|
|
log.Debug("Checking for update at: " + requestUriString);
|
|
var webRequest = WebRequest.Create(requestUriString);
|
|
webRequest.Timeout = 5000;
|
|
|
|
// Set the Method property of the request to POST.
|
|
webRequest.Method = "GET";
|
|
|
|
((HttpWebRequest)webRequest).IfModifiedSince = File.GetLastWriteTimeUtc(path);
|
|
|
|
// Get the response.
|
|
var response = webRequest.GetResponse();
|
|
// Display the status.
|
|
log.Debug("Response status: " + ((HttpWebResponse)response).StatusDescription);
|
|
// Get the stream containing content returned by the server.
|
|
//dataStream = response.GetResponseStream();
|
|
// Open the stream using a StreamReader for easy access.
|
|
|
|
bool updateFound = false;
|
|
|
|
if (File.Exists(path))
|
|
{
|
|
var fi = new FileInfo(path);
|
|
|
|
Version LocalVersion = new Version();
|
|
Version WebVersion = new Version();
|
|
|
|
if (File.Exists(path))
|
|
{
|
|
using (Stream fs = File.OpenRead(path))
|
|
{
|
|
using (StreamReader sr = new StreamReader(fs))
|
|
{
|
|
LocalVersion = new Version(sr.ReadLine());
|
|
sr.Close();
|
|
}
|
|
fs.Close();
|
|
}
|
|
}
|
|
|
|
using (StreamReader sr = new StreamReader(response.GetResponseStream()))
|
|
{
|
|
WebVersion = new Version(sr.ReadLine());
|
|
|
|
sr.Close();
|
|
}
|
|
|
|
|
|
|
|
log.Info("New file Check: local " + LocalVersion + " vs Remote " + WebVersion);
|
|
|
|
if (LocalVersion < WebVersion)
|
|
{
|
|
updateFound = true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
updateFound = true;
|
|
log.Info("File does not exist: Getting " + path);
|
|
// get it
|
|
}
|
|
|
|
response.Close();
|
|
|
|
if (updateFound)
|
|
{
|
|
var dr = CustomMessageBox.Show("Update Found\n\nDo you wish to update now?", "Update Now", MessageBoxButtons.YesNo);
|
|
if (dr == DialogResult.Yes)
|
|
{
|
|
DoUpdate();
|
|
}
|
|
else
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
public static void DoUpdate()
|
|
{
|
|
ProgressReporterDialogue frmProgressReporter = new ProgressReporterDialogue()
|
|
{
|
|
Text = "Check for Updates",
|
|
StartPosition = System.Windows.Forms.FormStartPosition.CenterScreen
|
|
};
|
|
|
|
ThemeManager.ApplyThemeTo(frmProgressReporter);
|
|
|
|
frmProgressReporter.DoWork += new Controls.ProgressReporterDialogue.DoWorkEventHandler(DoUpdateWorker_DoWork);
|
|
|
|
frmProgressReporter.UpdateProgressAndStatus(-1, "Checking for Updates");
|
|
|
|
frmProgressReporter.RunBackgroundOperationAsync();
|
|
}
|
|
|
|
static void DoUpdateWorker_DoWork(object sender, Controls.ProgressWorkerEventArgs e)
|
|
{
|
|
// TODO: Is this the right place?
|
|
#region Fetch Parameter Meta Data
|
|
|
|
var progressReporterDialogue = ((ProgressReporterDialogue)sender);
|
|
progressReporterDialogue.UpdateProgressAndStatus(-1, "Getting Updated Parameters");
|
|
|
|
try
|
|
{
|
|
ParameterMetaDataParser.GetParameterInformation();
|
|
}
|
|
catch (Exception ex) { log.Error(ex.ToString()); CustomMessageBox.Show("Error getting Parameter Information"); }
|
|
|
|
#endregion Fetch Parameter Meta Data
|
|
|
|
progressReporterDialogue.UpdateProgressAndStatus(-1, "Getting Base URL");
|
|
// check for updates
|
|
if (Debugger.IsAttached)
|
|
{
|
|
log.Info("Skipping update test as it appears we are debugging");
|
|
}
|
|
else
|
|
{
|
|
MainV2.updateCheckMain(progressReporterDialogue);
|
|
}
|
|
}
|
|
|
|
private static bool updateCheck(ProgressReporterDialogue frmProgressReporter, string baseurl, string subdir)
|
|
{
|
|
bool update = false;
|
|
List<string> files = new List<string>();
|
|
|
|
// Create a request using a URL that can receive a post.
|
|
log.Info(baseurl);
|
|
WebRequest request = WebRequest.Create(baseurl);
|
|
request.Timeout = 10000;
|
|
// Set the Method property of the request to POST.
|
|
request.Method = "GET";
|
|
// Get the request stream.
|
|
Stream dataStream; //= request.GetRequestStream();
|
|
// Get the response.
|
|
WebResponse response = request.GetResponse();
|
|
// Display the status.
|
|
log.Info(((HttpWebResponse)response).StatusDescription);
|
|
// Get the stream containing content returned by the server.
|
|
dataStream = response.GetResponseStream();
|
|
// Open the stream using a StreamReader for easy access.
|
|
StreamReader reader = new StreamReader(dataStream);
|
|
// Read the content.
|
|
string responseFromServer = reader.ReadToEnd();
|
|
// Display the content.
|
|
Regex regex = new Regex("href=\"([^\"]+)\"", RegexOptions.IgnoreCase);
|
|
if (regex.IsMatch(responseFromServer))
|
|
{
|
|
MatchCollection matchs = regex.Matches(responseFromServer);
|
|
for (int i = 0; i < matchs.Count; i++)
|
|
{
|
|
if (matchs[i].Groups[1].Value.ToString().Contains(".."))
|
|
continue;
|
|
if (matchs[i].Groups[1].Value.ToString().Contains("http"))
|
|
continue;
|
|
files.Add(System.Web.HttpUtility.UrlDecode(matchs[i].Groups[1].Value.ToString()));
|
|
}
|
|
}
|
|
|
|
//Console.WriteLine(responseFromServer);
|
|
// Clean up the streams.
|
|
reader.Close();
|
|
dataStream.Close();
|
|
response.Close();
|
|
|
|
string dir = Path.GetDirectoryName(Application.ExecutablePath) + Path.DirectorySeparatorChar + subdir;
|
|
if (!Directory.Exists(dir))
|
|
Directory.CreateDirectory(dir);
|
|
foreach (string file in files)
|
|
{
|
|
if (frmProgressReporter.doWorkArgs.CancelRequested)
|
|
{
|
|
frmProgressReporter.doWorkArgs.CancelAcknowledged = true;
|
|
throw new Exception("Cancel");
|
|
}
|
|
|
|
if (file.Equals("/"))
|
|
{
|
|
continue;
|
|
}
|
|
if (file.EndsWith("/"))
|
|
{
|
|
update = updateCheck(frmProgressReporter, baseurl + file, subdir.Replace('/', Path.DirectorySeparatorChar) + file) && update;
|
|
continue;
|
|
}
|
|
if (frmProgressReporter != null)
|
|
frmProgressReporter.UpdateProgressAndStatus(-1, "Checking " + file);
|
|
|
|
string path = Path.GetDirectoryName(Application.ExecutablePath) + Path.DirectorySeparatorChar + subdir + file;
|
|
|
|
|
|
// Create a request using a URL that can receive a post.
|
|
request = WebRequest.Create(baseurl + file);
|
|
log.Info(baseurl + file + " ");
|
|
// Set the Method property of the request to POST.
|
|
request.Method = "HEAD";
|
|
|
|
((HttpWebRequest)request).IfModifiedSince = File.GetLastWriteTimeUtc(path);
|
|
|
|
// Get the response.
|
|
response = request.GetResponse();
|
|
// Display the status.
|
|
log.Info(((HttpWebResponse)response).StatusDescription);
|
|
// Get the stream containing content returned by the server.
|
|
//dataStream = response.GetResponseStream();
|
|
// Open the stream using a StreamReader for easy access.
|
|
|
|
bool updateThisFile = false;
|
|
|
|
if (File.Exists(path))
|
|
{
|
|
FileInfo fi = new FileInfo(path);
|
|
|
|
//log.Info(response.Headers[HttpResponseHeader.ETag]);
|
|
string CurrentEtag = "";
|
|
|
|
if (File.Exists(path + ".etag"))
|
|
{
|
|
using (Stream fs = File.OpenRead(path + ".etag"))
|
|
{
|
|
using (StreamReader sr = new StreamReader(fs))
|
|
{
|
|
CurrentEtag = sr.ReadLine();
|
|
sr.Close();
|
|
}
|
|
fs.Close();
|
|
}
|
|
}
|
|
|
|
log.Debug("New file Check: " + fi.Length + " vs " + response.ContentLength + " " + response.Headers[HttpResponseHeader.ETag] + " vs " + CurrentEtag);
|
|
|
|
if (fi.Length != response.ContentLength || response.Headers[HttpResponseHeader.ETag] != CurrentEtag)
|
|
{
|
|
using (StreamWriter sw = new StreamWriter(path + ".etag.new"))
|
|
{
|
|
sw.WriteLine(response.Headers[HttpResponseHeader.ETag]);
|
|
sw.Close();
|
|
}
|
|
updateThisFile = true;
|
|
log.Info("NEW FILE " + file);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
updateThisFile = true;
|
|
log.Info("NEW FILE " + file);
|
|
// get it
|
|
}
|
|
|
|
reader.Close();
|
|
//dataStream.Close();
|
|
response.Close();
|
|
|
|
if (updateThisFile)
|
|
{
|
|
if (!update)
|
|
{
|
|
//DialogResult dr = MessageBox.Show("Update Found\n\nDo you wish to update now?", "Update Now", MessageBoxButtons.YesNo);
|
|
//if (dr == DialogResult.Yes)
|
|
{
|
|
update = true;
|
|
}
|
|
//else
|
|
{
|
|
// return;
|
|
}
|
|
}
|
|
if (frmProgressReporter != null)
|
|
frmProgressReporter.UpdateProgressAndStatus(-1, "Getting " + file);
|
|
|
|
// from head
|
|
long bytes = response.ContentLength;
|
|
|
|
// Create a request using a URL that can receive a post.
|
|
request = HttpWebRequest.Create(baseurl + file);
|
|
// Set the Method property of the request to POST.
|
|
request.Method = "GET";
|
|
|
|
((HttpWebRequest)request).AutomaticDecompression = DecompressionMethods.GZip | DecompressionMethods.Deflate;
|
|
|
|
request.Headers.Add("Accept-Encoding", "gzip,deflate");
|
|
|
|
// Get the response.
|
|
response = request.GetResponse();
|
|
// Display the status.
|
|
log.Debug(((HttpWebResponse)response).StatusDescription);
|
|
// Get the stream containing content returned by the server.
|
|
dataStream = response.GetResponseStream();
|
|
|
|
long contlen = bytes;
|
|
|
|
byte[] buf1 = new byte[1024];
|
|
|
|
FileStream fs = new FileStream(path + ".new", FileMode.Create); //
|
|
|
|
DateTime dt = DateTime.Now;
|
|
|
|
dataStream.ReadTimeout = 30000;
|
|
|
|
while (dataStream.CanRead)
|
|
{
|
|
try
|
|
{
|
|
if (dt.Second != DateTime.Now.Second)
|
|
{
|
|
if (frmProgressReporter != null)
|
|
frmProgressReporter.UpdateProgressAndStatus((int)(((double)(contlen - bytes) / (double)contlen) * 100), "Getting " + file + ": " + (((double)(contlen - bytes) / (double)contlen) * 100).ToString("0.0") + "%"); //+ Math.Abs(bytes) + " bytes");
|
|
dt = DateTime.Now;
|
|
}
|
|
}
|
|
catch { }
|
|
// log.Debug(file + " " + bytes);
|
|
int len = dataStream.Read(buf1, 0, 1024);
|
|
if (len == 0)
|
|
break;
|
|
bytes -= len;
|
|
fs.Write(buf1, 0, len);
|
|
}
|
|
|
|
fs.Close();
|
|
dataStream.Close();
|
|
response.Close();
|
|
}
|
|
|
|
|
|
}
|
|
|
|
//P.StartInfo.CreateNoWindow = true;
|
|
//P.StartInfo.RedirectStandardOutput = true;
|
|
return update;
|
|
|
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// trying to replicate google code etags....... this doesnt work.
|
|
/// </summary>
|
|
/// <param name="fileName"></param>
|
|
/// <param name="modifyDate"></param>
|
|
/// <returns></returns>
|
|
private string GetFileETag(string fileName, DateTime modifyDate)
|
|
{
|
|
|
|
string FileString;
|
|
|
|
System.Text.Encoder StringEncoder;
|
|
|
|
byte[] StringBytes;
|
|
|
|
MD5CryptoServiceProvider MD5Enc;
|
|
|
|
//use file name and modify date as the unique identifier
|
|
|
|
FileString = fileName + modifyDate.ToString("d", CultureInfo.InvariantCulture);
|
|
|
|
//get string bytes
|
|
|
|
StringEncoder = Encoding.UTF8.GetEncoder();
|
|
|
|
StringBytes = new byte[StringEncoder.GetByteCount(FileString.ToCharArray(), 0, FileString.Length, true)];
|
|
|
|
StringEncoder.GetBytes(FileString.ToCharArray(), 0, FileString.Length, StringBytes, 0, true);
|
|
|
|
//hash string using MD5 and return the hex-encoded hash
|
|
|
|
MD5Enc = new MD5CryptoServiceProvider();
|
|
|
|
byte[] hash = MD5Enc.ComputeHash((Stream)File.OpenRead(fileName));
|
|
|
|
return "\"" + BitConverter.ToString(hash).Replace("-", string.Empty) + "\"";
|
|
|
|
}
|
|
|
|
/// <summary>
|
|
/// keyboard shortcuts override
|
|
/// </summary>
|
|
/// <param name="msg"></param>
|
|
/// <param name="keyData"></param>
|
|
/// <returns></returns>
|
|
protected override bool ProcessCmdKey(ref Message msg, Keys keyData)
|
|
{
|
|
if (keyData == Keys.F12)
|
|
{
|
|
MenuConnect_Click(null, null);
|
|
return true;
|
|
}
|
|
|
|
if (keyData == (Keys.Control | Keys.F)) // temp
|
|
{
|
|
Form frm = new temp();
|
|
ThemeManager.ApplyThemeTo(frm);
|
|
frm.Show();
|
|
return true;
|
|
}
|
|
/*if (keyData == (Keys.Control | Keys.S)) // screenshot
|
|
{
|
|
ScreenShot();
|
|
return true;
|
|
}*/
|
|
if (keyData == (Keys.Control | Keys.G)) // nmea out
|
|
{
|
|
Form frm = new SerialOutput();
|
|
ThemeManager.ApplyThemeTo(frm);
|
|
frm.Show();
|
|
return true;
|
|
}
|
|
if (keyData == (Keys.Control | Keys.A)) // 3dr radio
|
|
{
|
|
Form temp = new Form();
|
|
Control frm = new _3DRradio();
|
|
temp.Controls.Add(frm);
|
|
temp.Size = frm.Size;
|
|
frm.Dock = DockStyle.Fill;
|
|
ThemeManager.ApplyThemeTo(temp);
|
|
temp.Show();
|
|
return true;
|
|
}
|
|
if (keyData == (Keys.Control | Keys.W)) // test ac config
|
|
{
|
|
|
|
Controls.ConfigPanel cfg = new Controls.ConfigPanel(Application.StartupPath + System.IO.Path.DirectorySeparatorChar + "ArduCopterConfig.xml");
|
|
|
|
//cfg.Show();
|
|
|
|
return true;
|
|
}
|
|
if (keyData == (Keys.Control | Keys.T)) // for override connect
|
|
{
|
|
try
|
|
{
|
|
MainV2.comPort.Open(false);
|
|
}
|
|
catch (Exception ex) { CustomMessageBox.Show(ex.ToString()); }
|
|
return true;
|
|
}
|
|
if (keyData == (Keys.Control | Keys.Y)) // for ryan beall
|
|
{
|
|
#if MAVLINK10
|
|
// write
|
|
MainV2.comPort.doCommand(MAVLink.MAV_CMD.PREFLIGHT_STORAGE, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f);
|
|
//read
|
|
///////MainV2.comPort.doCommand(MAVLink09.MAV_CMD.PREFLIGHT_STORAGE, 1.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f);
|
|
#else
|
|
MainV2.comPort.doAction(MAVLink.MAV_ACTION.MAV_ACTION_STORAGE_WRITE);
|
|
#endif
|
|
CustomMessageBox.Show("Done MAV_ACTION_STORAGE_WRITE");
|
|
return true;
|
|
}
|
|
if (keyData == (Keys.Control | Keys.J)) // for jani
|
|
{
|
|
string data = "!!";
|
|
Common.InputBox("inject", "enter data to be written", ref data);
|
|
MainV2.comPort.Write(data + "\r");
|
|
return true;
|
|
}
|
|
return base.ProcessCmdKey(ref msg, keyData);
|
|
}
|
|
|
|
public void changelanguage(CultureInfo ci)
|
|
{
|
|
if (ci != null && !Thread.CurrentThread.CurrentUICulture.Equals(ci))
|
|
{
|
|
Thread.CurrentThread.CurrentUICulture = ci;
|
|
config["language"] = ci.Name;
|
|
//System.Threading.Thread.CurrentThread.CurrentCulture = ci;
|
|
|
|
HashSet<Control> views = new HashSet<Control> { FlightData, FlightPlanner, Simulation };
|
|
|
|
foreach (Control view in MyView.Controls)
|
|
views.Add(view);
|
|
|
|
foreach (Control view in views)
|
|
{
|
|
if (view != null)
|
|
{
|
|
ComponentResourceManager rm = new ComponentResourceManager(view.GetType());
|
|
foreach (Control ctrl in view.Controls)
|
|
rm.ApplyResource(ctrl);
|
|
rm.ApplyResources(view, "$this");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private void MainV2_FormClosing(object sender, FormClosingEventArgs e)
|
|
{
|
|
config["MainHeight"] = this.Height;
|
|
config["MainWidth"] = this.Width;
|
|
config["MainMaximised"] = this.WindowState.ToString();
|
|
|
|
config["MainLocX"] = this.Location.X.ToString();
|
|
config["MainLocY"] = this.Location.Y.ToString();
|
|
|
|
try
|
|
{
|
|
comPort.logreadmode = false;
|
|
if (comPort.logfile != null)
|
|
comPort.logfile.Close();
|
|
|
|
if (comPort.rawlogfile != null)
|
|
comPort.rawlogfile.Close();
|
|
|
|
comPort.logfile = null;
|
|
comPort.rawlogfile = null;
|
|
}
|
|
catch { }
|
|
|
|
}
|
|
|
|
public static string getConfig(string paramname)
|
|
{
|
|
if (config[paramname] != null)
|
|
return config[paramname].ToString();
|
|
return "";
|
|
}
|
|
|
|
public void ChangeUnits()
|
|
{
|
|
try
|
|
{
|
|
// dist
|
|
if (MainV2.config["distunits"] != null)
|
|
{
|
|
switch ((Common.distances)Enum.Parse(typeof(Common.distances), MainV2.config["distunits"].ToString()))
|
|
{
|
|
case Common.distances.Meters:
|
|
MainV2.comPort.MAV.cs.multiplierdist = 1;
|
|
MainV2.comPort.MAV.cs.DistanceUnit = "Meters";
|
|
break;
|
|
case Common.distances.Feet:
|
|
MainV2.comPort.MAV.cs.multiplierdist = 3.2808399f;
|
|
MainV2.comPort.MAV.cs.DistanceUnit = "Feet";
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
MainV2.comPort.MAV.cs.multiplierdist = 1;
|
|
MainV2.comPort.MAV.cs.DistanceUnit = "Meters";
|
|
}
|
|
|
|
// speed
|
|
if (MainV2.config["speedunits"] != null)
|
|
{
|
|
switch ((Common.speeds)Enum.Parse(typeof(Common.speeds), MainV2.config["speedunits"].ToString()))
|
|
{
|
|
case Common.speeds.ms:
|
|
MainV2.comPort.MAV.cs.multiplierspeed = 1;
|
|
MainV2.comPort.MAV.cs.SpeedUnit = "m/s";
|
|
break;
|
|
case Common.speeds.fps:
|
|
MainV2.comPort.MAV.cs.multiplierdist = 3.2808399f;
|
|
MainV2.comPort.MAV.cs.SpeedUnit = "fps";
|
|
break;
|
|
case Common.speeds.kph:
|
|
MainV2.comPort.MAV.cs.multiplierspeed = 3.6f;
|
|
MainV2.comPort.MAV.cs.SpeedUnit = "kph";
|
|
break;
|
|
case Common.speeds.mph:
|
|
MainV2.comPort.MAV.cs.multiplierspeed = 2.23693629f;
|
|
MainV2.comPort.MAV.cs.SpeedUnit = "mph";
|
|
break;
|
|
case Common.speeds.knots:
|
|
MainV2.comPort.MAV.cs.multiplierspeed = 1.94384449f;
|
|
MainV2.comPort.MAV.cs.SpeedUnit = "knots";
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
MainV2.comPort.MAV.cs.multiplierspeed = 1;
|
|
MainV2.comPort.MAV.cs.SpeedUnit = "m/s";
|
|
}
|
|
}
|
|
catch { }
|
|
|
|
}
|
|
|
|
private void CMB_baudrate_TextChanged(object sender, EventArgs e)
|
|
{
|
|
var sb = new StringBuilder();
|
|
int baud = 0;
|
|
for (int i = 0; i < _connectionControl.CMB_baudrate.Text.Length; i++)
|
|
if (char.IsDigit(_connectionControl.CMB_baudrate.Text[i]))
|
|
{
|
|
sb.Append(_connectionControl.CMB_baudrate.Text[i]);
|
|
baud = baud * 10 + _connectionControl.CMB_baudrate.Text[i] - '0';
|
|
}
|
|
if (_connectionControl.CMB_baudrate.Text != sb.ToString())
|
|
{
|
|
_connectionControl.CMB_baudrate.Text = sb.ToString();
|
|
}
|
|
try
|
|
{
|
|
if (baud > 0 && comPort.BaseStream.BaudRate != baud)
|
|
comPort.BaseStream.BaudRate = baud;
|
|
}
|
|
catch (Exception)
|
|
{
|
|
}
|
|
}
|
|
|
|
private void CMB_serialport_Enter(object sender, EventArgs e)
|
|
{
|
|
CMB_serialport_Click(sender, e);
|
|
}
|
|
|
|
private void MainMenu_MouseLeave(object sender, EventArgs e)
|
|
{
|
|
if (_connectionControl.PointToClient(Control.MousePosition).Y < MainMenu.Height)
|
|
return;
|
|
|
|
this.SuspendLayout();
|
|
|
|
panel1.Visible = false;
|
|
|
|
this.ResumeLayout();
|
|
}
|
|
|
|
void menu_MouseEnter(object sender, EventArgs e)
|
|
{
|
|
this.SuspendLayout();
|
|
panel1.Location = new Point(0,0);
|
|
panel1.Width = menu.Width;
|
|
panel1.BringToFront();
|
|
panel1.Visible = true;
|
|
this.ResumeLayout();
|
|
}
|
|
|
|
private void autoHideToolStripMenuItem_Click(object sender, EventArgs e)
|
|
{
|
|
AutoHideMenu(autoHideToolStripMenuItem.Checked);
|
|
|
|
config["menu_autohide"] = autoHideToolStripMenuItem.Checked.ToString();
|
|
}
|
|
|
|
void AutoHideMenu(bool hide)
|
|
{
|
|
autoHideToolStripMenuItem.Checked = hide;
|
|
|
|
if (!hide)
|
|
{
|
|
this.SuspendLayout();
|
|
panel1.Dock = DockStyle.Top;
|
|
panel1.SendToBack();
|
|
panel1.Visible = true;
|
|
menu.Visible = false;
|
|
MainMenu.MouseLeave -= MainMenu_MouseLeave;
|
|
panel1.MouseLeave -= MainMenu_MouseLeave;
|
|
toolStripConnectionControl.MouseLeave -= MainMenu_MouseLeave;
|
|
this.ResumeLayout();
|
|
}
|
|
else
|
|
{
|
|
this.SuspendLayout();
|
|
panel1.Dock = DockStyle.None;
|
|
panel1.Visible = false;
|
|
MainMenu.MouseLeave += MainMenu_MouseLeave;
|
|
panel1.MouseLeave += MainMenu_MouseLeave;
|
|
toolStripConnectionControl.MouseLeave += MainMenu_MouseLeave;
|
|
menu.Visible = true;
|
|
menu.SendToBack();
|
|
this.ResumeLayout();
|
|
}
|
|
}
|
|
}
|
|
} |