前文 由於經常用到串口調試, 儘管有現成的軟體, 因為前端時間涉及一個二次開發, 就因為一個RtsEnable設置, 折騰半天, 網上各種版本的也很多, 功能擴展的很開也多。所以現在自己做了一個夠用版,基於自己的需求,簡單的實現發送接收功能, 至於那些擴展功能可以自己根據需求添加。 正文 先上個運行 ...
前文
由於經常用到串口調試, 儘管有現成的軟體, 因為前端時間涉及一個二次開發, 就因為一個RtsEnable設置, 折騰半天, 網上各種版本的也很多, 功能擴展的很開也多。所以現在自己做了一個夠用版,基於自己的需求,簡單的實現發送接收功能, 至於那些擴展功能可以自己根據需求添加。
正文
先上個運行效果圖:
項目架構
該實例用的GalaSoft.Mvvm, 該插件可以直接在NuGet中並且添加。
1.串口參數 , 為了方便, 埠號並沒有用動態載入的方式, 如下枚舉結構:
/// <summary> /// 埠號 /// </summary> public enum Port { COM1, COM2, COM3, COM4, COM5, COM6, COM7, COM8, COM9, COM10, COM11, COM12, COM13, COM14, COM15, COM16, COM17, COM18, COM19, COM20, COM21, COM22, COM23, COM24, COM25, COM26, COM27, COM28, COM29, COM30 } /// <summary> /// 奇偶校驗 /// </summary> public enum CheckMode { None = 0, Odd = 1, Even = 2, Mark = 3, Space = 4 } /// <summary> /// 停止位 /// </summary> public enum StopBit { One=1, Two=2, OnePointFive=3, }
2.串口參數配置類 ,
作用: 主要用於綁定界面的參數選項。
/// <summary> /// 串口參數設置類 /// </summary> public class ComParameterConfig : ViewModelBase { public ComParameterConfig() { Port = System.Enum.GetValues(typeof(Port)); CheckMode = System.Enum.GetValues(typeof(CheckMode)); StopBit = System.Enum.GetValues(typeof(StopBit)); BaudRate = new List<int>() { 110, 300, 600, 1200, 2400, 4800, 9600, 14400, 19200, 38400, 56000, 57600, 115200, }; DataBit = new List<int>() { 6, 7, 8 }; } private Array port; private Array checkMode; private Array stopBit; private List<int> dataBit; private List<int> baudRate; /// <summary> /// 埠 /// </summary> public Array Port { get { return port; } set { port = value; RaisePropertyChanged(); } } /// <summary> /// 校驗模式 /// </summary> public Array CheckMode { get { return checkMode; } set { checkMode = value; RaisePropertyChanged(); } } /// <summary> /// 停止位 /// </summary> public Array StopBit { get { return stopBit; } set { stopBit = value; RaisePropertyChanged(); } } /// <summary> /// 波特率 /// </summary> public List<int> BaudRate { get { return baudRate; } set { baudRate = value; RaisePropertyChanged(); } } /// <summary> /// 數據位 /// </summary> public List<int> DataBit { get { return dataBit; } set { dataBit = value; RaisePropertyChanged(); } } }
3.當前配置參數類
作用: 用於保存當前的串口參數、串口功能開關接收數據等業務。
/// <summary> /// 當前配置參數 /// </summary> public class CurrentParameter : ViewModelBase { private int baudRdate = 9600; private int dataBit = 8; private Port port; private CheckMode checkMode; private StopBit stopBit = StopBit.One; private SerialPort serialPort; private string dataReceiveInfo; private string sendData; private bool isOpen; #region UI綁定參數 /// <summary> /// 接收區數據 /// </summary> public string DataReceiveInfo { get { return dataReceiveInfo; } set { dataReceiveInfo = value; RaisePropertyChanged(); } } /// <summary> /// 發送數據 /// </summary> public string SendData { get { return sendData; } set { sendData = value; RaisePropertyChanged(); } } #endregion #region 串口參數信息 /// <summary> /// 開關 /// </summary> public bool IsOpen { get { return isOpen; } set { isOpen = value; RaisePropertyChanged(); } } /// <summary> /// 數據位 /// </summary> public int DataBit { get { return dataBit; } set { dataBit = value; RaisePropertyChanged(); } } /// <summary> /// 波特率 /// </summary> public int BaudRdate { get { return baudRdate; } set { baudRdate = value; RaisePropertyChanged(); } } /// <summary> /// 埠 /// </summary> public Port Port { get { return port; } set { port = value; RaisePropertyChanged(); } } /// <summary> /// 校驗 /// </summary> public CheckMode CheckMode { get { return checkMode; } set { checkMode = value; RaisePropertyChanged(); } } /// <summary> /// 停止位 /// </summary> public StopBit StopBit { get { return stopBit; } set { stopBit = value; RaisePropertyChanged(); } } /// <summary> /// COM /// </summary> public SerialPort SerialPort { get { return serialPort; } set { serialPort = value; RaisePropertyChanged(); } } #endregion #region 串口操作方法 /// <summary> /// 開啟串口 /// </summary> /// <returns></returns> public bool Open() { if (serialPort != null && serialPort.IsOpen) { return Close(); } try { serialPort = new SerialPort(); serialPort.DataBits = this.DataBit; serialPort.StopBits = ComHelper.GetStopBits(this.StopBit.ToString()); serialPort.Parity = ComHelper.GetParity(this.CheckMode.ToString()); serialPort.PortName = this.Port.ToString(); serialPort.RtsEnable = true; serialPort.DataReceived += SerialPort_DataReceived; serialPort.Open(); if (serialPort.IsOpen) return IsOpen = true; else return IsOpen = false; } catch (Exception ex) { MessageBox.Show(ex.Message); } return IsOpen = false; } /// <summary> /// 關閉串口 /// </summary> /// <returns></returns> public bool Close() { try { if (serialPort.IsOpen) { serialPort.Close(); return IsOpen = serialPort.IsOpen; } else { return IsOpen = serialPort.IsOpen; } } catch (Exception ex) { MessageBox.Show(ex.Message); return IsOpen = false; } } /// <summary> /// 發送數據 /// </summary> public void Send() { } #endregion /// <summary> /// 返回事件 /// </summary> /// <param name="sender"></param> /// <param name="e"></param> private void SerialPort_DataReceived(object sender, SerialDataReceivedEventArgs e) { Thread.Sleep(100); string ReadText = serialPort.ReadExisting(); DataReceiveInfo += ReadText + "\r\n"; } }
4.核心MainViewModel類
作用: 關聯首頁的上下文, 通過DataContext綁定, 關聯界面元素、命令等作用。
public class MainViewModel : ViewModelBase { /// <summary> /// Initializes a new instance of the MainViewModel class. /// </summary> public MainViewModel() { ComParameterConfig = new ComParameterConfig(); CurrentParameter = new CurrentParameter(); } private ComParameterConfig comParameter; /// <summary> /// 參數類 /// </summary> public ComParameterConfig ComParameterConfig { get { return comParameter; } set { comParameter = value; RaisePropertyChanged(); } } private CurrentParameter currentParameter; /// <summary> /// 當前配置參數 /// </summary> public CurrentParameter CurrentParameter { get { return currentParameter; } set { currentParameter = value; RaisePropertyChanged(); } } #region Command private RelayCommand _ToOpen; public RelayCommand ToOpen { get { if (_ToOpen == null) { _ToOpen = new RelayCommand(Open); } return _ToOpen; } set { _ToOpen = value; } } /// <summary> /// 根據配置打開埠 /// </summary> public void Open() { this.CurrentParameter.Open(); } private RelayCommand _ToClick; public RelayCommand ToClick { get { if (_ToClick == null) { _ToClick = new RelayCommand(Click); } return _ToClick; } set { _ToClick = value; } } /// <summary> /// 發送數據 /// </summary> public void Click() { this.currentParameter.Send(); } #endregion }
5.CRC校驗核心類
作用:主要實現數據校驗, 含ModbusCR標準校驗
/// <summary> /// CRC校驗 /// </summary> public class CRC { #region CRC16 public static byte[] CRC16(byte[] data) { int len = data.Length; if (len > 0) { ushort crc = 0xFFFF; for (int i = 0; i < len; i++) { crc = (ushort)(crc ^ (data[i])); for (int j = 0; j < 8; j++) { crc = (crc & 1) != 0 ? (ushort)((crc >> 1) ^ 0xA001) : (ushort)(crc >> 1); } } byte hi = (byte)((crc & 0xFF00) >> 8); //高位置 byte lo = (byte)(crc & 0x00FF); //低位置 return new byte[] { hi, lo }; } return new byte[] { 0, 0 }; } #endregion #region ToCRC16 public static string ToCRC16(string content) { return ToCRC16(content, Encoding.UTF8); } public static string ToCRC16(string content, bool isReverse) { return ToCRC16(content, Encoding.UTF8, isReverse); } public static string ToCRC16(string content, Encoding encoding) { return ByteToString(CRC16(encoding.GetBytes(content)), true); } public static string ToCRC16(string content, Encoding encoding, bool isReverse) { return ByteToString(CRC16(encoding.GetBytes(content)), isReverse); } public static string ToCRC16(byte[] data) { return ByteToString(CRC16(data), true); } public static string ToCRC16(byte[] data, bool isReverse) { return ByteToString(CRC16(data), isReverse); } #endregion #region ToModbusCRC16 public static string ToModbusCRC16(string s) { return ToModbusCRC16(s, true); } public static string ToModbusCRC16(string s, bool isReverse) { return ByteToString(CRC16(StringToHexByte(s)), isReverse); } public static string ToModbusCRC16(byte[] data) { return ToModbusCRC16(data, true); } public static string ToModbusCRC16(byte[] data, bool isReverse) { return ByteToString(CRC16(data), isReverse); } #endregion #region ByteToString public static string ByteToString(byte[] arr, bool isReverse) { try { byte hi = arr[0], lo = arr[1]; return Convert.ToString(isReverse ? hi + lo * 0x100 : hi * 0x100 + lo, 16).ToUpper().PadLeft(4, '0'); } catch (Exception ex) { throw (ex); } } public static string ByteToString(byte[] arr) { try { return ByteToString(arr, true); } catch (Exception ex) { throw (ex); } } #endregion #region StringToHexString public static string StringToHexString(string str) { StringBuilder s = new StringBuilder(); foreach (short c in str.ToCharArray()) { s.Append(c.ToString("X4")); } return s.ToString(); } #endregion #region StringToHexByte private static string ConvertChinese(string str) { StringBuilder s = new StringBuilder(); foreach (short c in str.ToCharArray()) { if (c <= 0 || c >= 127) { s.Append(c.ToString("X4")); } else { s.Append((char)c); } } return s.ToString(); } private static string FilterChinese(string str) { StringBuilder s = new StringBuilder(); foreach (short c in str.ToCharArray()) { if (c > 0 && c < 127) { s.Append((char)c); } } return s.ToString(); } /// <summary> /// 字元串轉16進位字元數組 /// </summary> /// <param name="hex"></param> /// <returns></returns> public static byte[] StringToHexByte(string str) { return StringToHexByte(str, false); } /// <summary> /// 字元串轉16進位字元數組 /// </summary> /// <param name="str"></param> /// <param name="isFilterChinese">是否過濾掉中文字元</param> /// <returns></returns> public static byte[] StringToHexByte(string str, bool isFilterChinese) { string hex = isFilterChinese ? FilterChinese(str) : ConvertChinese(str); //清除所有空格 hex = hex.Replace(" ", ""); //若字元個數為奇數,補一個0 hex += hex.Length % 2 != 0 ? "0" : ""; byte[] result = new byte[hex.Length / 2]; for (int i = 0, c = result.Length; i < c; i++) { result[i] = Convert.ToByte(hex.Substring(i * 2, 2), 16); } return result; } #endregion }
WPF技術點:
1.自定義樣式按鈕
<Style x:Key="CommonButtonBase" TargetType="{x:Type Button}"> <Setter Property="BorderBrush" Value="Transparent"/> <Setter Property="BorderThickness" Value="0"/> <Setter Property="Foreground" Value="{DynamicResource {x:Static SystemColors.ControlTextBrushKey}}"/> <Setter Property="HorizontalContentAlignment" Value="Center"/> <Setter Property="VerticalContentAlignment" Value="Center"/> <Setter Property="Padding" Value="1"/> <Setter Property="Cursor" Value="Hand"/> <Setter Property="Template"> <Setter.Value> <ControlTemplate TargetType="{x:Type Button}"> <Border x:Name="border" BorderBrush="{TemplateBinding BorderBrush}" CornerRadius="4" BorderThickness="{TemplateBinding BorderThickness}" Background="{TemplateBinding Background}" SnapsToDevicePixels="true"> <ContentPresenter HorizontalAlignment="{TemplateBinding HorizontalContentAlignment}" Margin="{TemplateBinding Padding}" RecognizesAccessKey="True" SnapsToDevicePixels="{TemplateBinding SnapsToDevicePixels}" VerticalAlignment="{TemplateBinding VerticalContentAlignment}"/> </Border> <ControlTemplate.Triggers> <Trigger Property="IsEnabled" Value="False"> <Setter Property="Background" Value="#000000"/> <Setter Property="Opacity" Value="0.1"/> </Trigger> <Trigger Property="IsMouseOver" Value="true"> <Setter Property="Foreground" Value="#FFFF00"/> </Trigger> <Trigger Property="IsMouseOver" Value="false"> <Setter Property="Foreground" Value="White"/> </Trigger> </ControlTemplate.Triggers> </ControlTemplate> </Setter.Value> </Setter> </Style> <Style TargetType="{x:Type Button}" x:Key="Btn0093EABase" BasedOn="{StaticResource CommonButtonBase}"> <Setter Property="Background" Value="#0093EA"/> <Setter Property="Foreground" Value="White"/> <Setter Property="FontSize" Value="22"/> <Setter Property="Height" Value="40"/> <Setter Property="Margin" Value="5"/> </Style>
2.轉換器用於綁定按鈕
public class FontConverters : IValueConverter { public object Convert(object value, Type targetType, object parameter, CultureInfo culture) { if (value != null && bool.TryParse(value.ToString(), out bool result)) { if (result) { return "關閉串口"; } } return "打開串口"; } public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture) { throw new NotImplementedException(); } }
//用於綁定UI的顏色狀態顯示
public class ColorConverters : IValueConverter { public object Convert(object value, Type targetType, object parameter, CultureInfo culture) { if (value != null && bool.TryParse(value.ToString(), out bool result)) { if (result) { return new SolidColorBrush((Color)System.Windows.Media.ColorConverter.ConvertFromString("#2E8B57")); } } return new SolidColorBrush((Color)System.Windows.Media.ColorConverter.ConvertFromString("#FF6347")); } public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture) { throw new NotImplementedException(); } }
3.引用字體
<TextBlock Text="" Margin="20 5 0 5" FontFamily="pack://application:,,,/Font/#iconfont" Foreground="White" FontSize="30" VerticalAlignment="Center"/>
4.綁定命令和元素
<TextBlock Text="端 口:" Style="{DynamicResource TxtComStyle}"/> <ComboBox Grid.Row="0" Grid.Column="2" Style="{StaticResource ComboBoxStyle}" SelectedItem="{Binding CurrentParameter.Port}" ItemsSource="{Binding ComParameterConfig.Port}" /> <TextBlock Text="波 特 率:" Style="{DynamicResource TxtComStyle}"/> <ComboBox Grid.Row="0" Grid.Column="2" Style="{StaticResource ComboBoxStyle}" SelectedItem