diff --git a/WebSocketTool.sln b/WebSocketTool.sln new file mode 100644 index 0000000..5a5dcba --- /dev/null +++ b/WebSocketTool.sln @@ -0,0 +1,25 @@ + +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio Version 17 +VisualStudioVersion = 17.0.31912.275 +MinimumVisualStudioVersion = 10.0.40219.1 +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "WebSocketTool", "WebSocketTool\WebSocketTool.csproj", "{1002416D-BE37-48F3-97FA-630D37C5FC91}" +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|Any CPU = Debug|Any CPU + Release|Any CPU = Release|Any CPU + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {1002416D-BE37-48F3-97FA-630D37C5FC91}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {1002416D-BE37-48F3-97FA-630D37C5FC91}.Debug|Any CPU.Build.0 = Debug|Any CPU + {1002416D-BE37-48F3-97FA-630D37C5FC91}.Release|Any CPU.ActiveCfg = Release|Any CPU + {1002416D-BE37-48F3-97FA-630D37C5FC91}.Release|Any CPU.Build.0 = Release|Any CPU + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection + GlobalSection(ExtensibilityGlobals) = postSolution + SolutionGuid = {DBC3AE78-8E92-460A-9FE2-0FA2D754C735} + EndGlobalSection +EndGlobal diff --git a/WebSocketTool.sln.DotSettings b/WebSocketTool.sln.DotSettings new file mode 100644 index 0000000..981fcca --- /dev/null +++ b/WebSocketTool.sln.DotSettings @@ -0,0 +1,2 @@ + + True \ No newline at end of file diff --git a/WebSocketTool/App.config b/WebSocketTool/App.config new file mode 100644 index 0000000..88fa402 --- /dev/null +++ b/WebSocketTool/App.config @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/WebSocketTool/App.xaml b/WebSocketTool/App.xaml new file mode 100644 index 0000000..e1c717d --- /dev/null +++ b/WebSocketTool/App.xaml @@ -0,0 +1,9 @@ + + + + + diff --git a/WebSocketTool/App.xaml.cs b/WebSocketTool/App.xaml.cs new file mode 100644 index 0000000..f77fda6 --- /dev/null +++ b/WebSocketTool/App.xaml.cs @@ -0,0 +1,45 @@ +using System; +using System.Collections.Generic; +using System.Configuration; +using System.Data; +using System.IO; +using System.Linq; +using System.Threading.Tasks; +using System.Windows; +using log4net; +using log4net.Config; + +namespace WebSocketTool +{ + /// + /// App.xaml 的交互逻辑 + /// + public partial class App : Application + { + public static readonly ILog Log = LogManager.GetLogger(nameof(App)); + + protected override void OnStartup(StartupEventArgs e) + { + base.OnStartup(e); + XmlConfigurator.ConfigureAndWatch(new FileInfo("log4net.config")); + Directory.CreateDirectory("log"); + } + + public static void RunOnUIThread(Action action) + { + if (Current == null) + { + Log.Info("application current is null"); + return; + } + if (Current.CheckAccess()) + { + action?.Invoke(); + } + else + { + Current.Dispatcher.BeginInvoke(action); + } + } + } +} diff --git a/WebSocketTool/Base/BaseViewModel.cs b/WebSocketTool/Base/BaseViewModel.cs new file mode 100644 index 0000000..5138605 --- /dev/null +++ b/WebSocketTool/Base/BaseViewModel.cs @@ -0,0 +1,22 @@ +using System.ComponentModel; +using System.Runtime.CompilerServices; +using WebSocketTool.Annotations; + +namespace WebSocketTool.Base +{ + public class BaseViewModel : ObservableObject + { + + } + + public class ObservableObject : INotifyPropertyChanged + { + public event PropertyChangedEventHandler PropertyChanged; + + [NotifyPropertyChangedInvocator] + protected virtual void RaisePropertyChanged([CallerMemberName] string propertyName = null) + { + PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName)); + } + } +} \ No newline at end of file diff --git a/WebSocketTool/Client/ClientViewModel.cs b/WebSocketTool/Client/ClientViewModel.cs new file mode 100644 index 0000000..a562c99 --- /dev/null +++ b/WebSocketTool/Client/ClientViewModel.cs @@ -0,0 +1,240 @@ +using System; +using System.Windows.Threading; +using log4net; +using WebSocketSharp; +using WebSocketTool.Base; + +namespace WebSocketTool.Client +{ + public class ClientViewModel : BaseViewModel + { + private static readonly ILog Log = LogManager.GetLogger(nameof(ClientViewModel)); + private IClientView view; + private SocketClient mClient; + + public ClientViewModel(IClientView view) + { + this.view = view; + } + + private string mWsUrl; + public string WsUrl + { + get => mWsUrl; + set + { + mWsUrl = value; + RaisePropertyChanged(nameof(WsUrl)); + } + } + + private string mSendContent = string.Empty; + public string SendContent + { + get => mSendContent; + set + { + mSendContent = value; + RaisePropertyChanged(nameof(SendContent)); + } + } + + private long mPingTime = 1000; + public long PingTime + { + get => mPingTime; + set + { + mPingTime = value; + RaisePropertyChanged(nameof(PingTime)); + } + } + + public void Connect() + { + if (string.IsNullOrEmpty(WsUrl)) + { + view.AppendInfo("请输入正确的WebSocket地址"); + return; + } + + if (mClient == null) + { + InitSocketClient(); + } + + if (mClient.IsAlive()) + { + view.AppendInfo("WebSocket已连接,请先断开上次连接"); + return; + } + + view.AppendInfo($"<=== start connect"); + mClient.ConnectAsync(); + } + + public void Send() + { + view.AppendInfo($"<=== socket send:{SendContent}"); + mClient.Send(SendContent); + } + + private bool mIsAlive = false; + + private bool mIsConnectEnable = true; + public bool IsConnectEnable + { + get => mIsConnectEnable; + set + { + mIsConnectEnable = value; + RaisePropertyChanged(nameof(IsConnectEnable)); + } + } + + private bool mIsCloseEnable = false; + public bool IsCloseEnable + { + get => mIsCloseEnable; + set + { + mIsCloseEnable = value; + RaisePropertyChanged(nameof(IsCloseEnable)); + } + } + + private bool mIsPingChecked; + public bool IsPingChecked + { + get => mIsPingChecked; + set + { + mIsPingChecked = value; + RaisePropertyChanged(nameof(IsPingChecked)); + } + } + + private bool mIsPingEnable; + public bool IsPingEnable + { + get => mIsPingEnable; + set + { + mIsPingEnable = value; + RaisePropertyChanged(nameof(IsPingEnable)); + } + } + + public void Close() + { + view.AppendInfo($"<=== start close"); + mClient?.CloseAsync(); + } + + private DispatcherTimer pingTimer; + public void StartPing() + { + if (!mIsAlive) + { + view.AppendInfo("start ping failure: ws is not connected"); + return; + } + if (pingTimer?.IsEnabled ?? false) + { + pingTimer.Stop(); + } + pingTimer = new DispatcherTimer(); + pingTimer.Interval = TimeSpan.FromMilliseconds(PingTime); + pingTimer.Tick += (sender, args) => + { + Ping(); + }; + pingTimer.Start(); + view.AppendInfo($"<===StartPing, time:{PingTime}"); + } + + private void Ping(string msg = null) + { + view.AppendInfo($"<=== ping:{msg}"); + mClient.Ping(msg); + } + + public void StopPing() + { + view.AppendInfo("<===StopPing"); + if (pingTimer?.IsEnabled ?? false) + { + pingTimer.Stop(); + pingTimer = null; + } + } + + private void InitSocketClient() + { + view.AppendInfo("InitSocketClient"); + mClient = new SocketClient(WsUrl); + mClient.OpenEvent += ClientOnOpenEvent; + mClient.CloseEvent += ClientOnCloseEvent; + mClient.ErrorEvent += ClientOnErrorEvent; + mClient.MessageEvent += ClientOnMessageEvent; + } + + private void ClientOnMessageEvent(object sender, MessageEventArgs e) + { + App.RunOnUIThread(() => + { + view.AppendInfo($"===> receive message: {e.Data}"); + }); + } + + private void ClientOnErrorEvent(object sender, ErrorEventArgs e) + { + App.RunOnUIThread(() => + { + view.AppendInfo($"===> socket error: {e.Message}"); + SetState(false); + }); + StopPing(); + } + + private void SetState(bool isAlive) + { + mIsAlive = isAlive; + if (isAlive) + { + IsPingEnable = true; + IsConnectEnable = false; + IsCloseEnable = true; + } + else + { + IsPingEnable = false; + if (IsPingChecked) + { + IsPingChecked = false; + } + IsConnectEnable = true; + IsCloseEnable = false; + } + } + + private void ClientOnCloseEvent(object sender, CloseEventArgs e) + { + App.RunOnUIThread(() => + { + view.AppendInfo($"===> socket closed:{e.Code}:{e.Reason}"); + SetState(false); + }); + StopPing(); + } + + private void ClientOnOpenEvent(object sender, EventArgs e) + { + App.RunOnUIThread(() => + { + view.AppendInfo($"<=== socket connected"); + SetState(true); + }); + } + } +} \ No newline at end of file diff --git a/WebSocketTool/Client/ClientWindow.xaml b/WebSocketTool/Client/ClientWindow.xaml new file mode 100644 index 0000000..7e9317c --- /dev/null +++ b/WebSocketTool/Client/ClientWindow.xaml @@ -0,0 +1,60 @@ + + + + + + + + + + + + + + +