@@ -1,6 +1,6 @@ | |||||
| | ||||
Microsoft Visual Studio Solution File, Format Version 12.00 | Microsoft Visual Studio Solution File, Format Version 12.00 | ||||
# Visual Studio Express 2012 for Windows Desktop | |||||
# Visual Studio 2012 | |||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "shadowsocks-csharp", "shadowsocks-csharp\shadowsocks-csharp.csproj", "{8C02D2F7-7CDB-4D55-9F25-CD03EF4AA062}" | Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "shadowsocks-csharp", "shadowsocks-csharp\shadowsocks-csharp.csproj", "{8C02D2F7-7CDB-4D55-9F25-CD03EF4AA062}" | ||||
EndProject | EndProject | ||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "test", "test\test.csproj", "{45913187-0685-4903-B250-DCEF0479CD86}" | Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "test", "test\test.csproj", "{45913187-0685-4903-B250-DCEF0479CD86}" | ||||
@@ -10,9 +10,9 @@ namespace Shadowsocks.Controller | |||||
{ | { | ||||
class Local | class Local | ||||
{ | |||||
{ | |||||
private Server _server; | private Server _server; | ||||
private bool _shareOverLAN; | |||||
private bool _shareOverLAN; | |||||
//private Encryptor encryptor; | //private Encryptor encryptor; | ||||
Socket _listener; | Socket _listener; | ||||
public Local(Configuration config) | public Local(Configuration config) | ||||
@@ -26,7 +26,7 @@ namespace Shadowsocks.Controller | |||||
{ | { | ||||
try | try | ||||
{ | { | ||||
// Create a TCP/IP socket. | |||||
// Create a TCP/IP socket. | |||||
_listener = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp); | _listener = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp); | ||||
_listener.SetSocketOption(SocketOptionLevel.Socket, SocketOptionName.ReuseAddress, true); | _listener.SetSocketOption(SocketOptionLevel.Socket, SocketOptionName.ReuseAddress, true); | ||||
IPEndPoint localEndPoint = null; | IPEndPoint localEndPoint = null; | ||||
@@ -36,7 +36,7 @@ namespace Shadowsocks.Controller | |||||
} | } | ||||
else | else | ||||
{ | { | ||||
localEndPoint = new IPEndPoint(IPAddress.Loopback, _server.local_port); | |||||
localEndPoint = new IPEndPoint(IPAddress.Loopback, _server.local_port); | |||||
} | } | ||||
// Bind the socket to the local endpoint and listen for incoming connections. | // Bind the socket to the local endpoint and listen for incoming connections. | ||||
@@ -224,7 +224,7 @@ namespace Shadowsocks.Controller | |||||
// reject socks 4 | // reject socks 4 | ||||
response = new byte[]{ 0, 91 }; | response = new byte[]{ 0, 91 }; | ||||
} | } | ||||
connection.BeginSend(response, 0, response.Length, 0, new AsyncCallback(handshakeSendCallback), null); | |||||
connection.BeginSend(response, 0, response.Length, 0, new AsyncCallback(HandshakeSendCallback), null); | |||||
} | } | ||||
else | else | ||||
{ | { | ||||
@@ -238,7 +238,7 @@ namespace Shadowsocks.Controller | |||||
} | } | ||||
} | } | ||||
private void handshakeSendCallback(IAsyncResult ar) | |||||
private void HandshakeSendCallback(IAsyncResult ar) | |||||
{ | { | ||||
try | try | ||||
{ | { | ||||
@@ -270,7 +270,7 @@ namespace Shadowsocks.Controller | |||||
if (bytesRead > 0) | if (bytesRead > 0) | ||||
{ | { | ||||
byte[] response = { 5, 0, 0, 1, 0, 0, 0, 0, 0, 0 }; | byte[] response = { 5, 0, 0, 1, 0, 0, 0, 0, 0, 0 }; | ||||
connection.BeginSend(response, 0, response.Length, 0, new AsyncCallback(startPipe), null); | |||||
connection.BeginSend(response, 0, response.Length, 0, new AsyncCallback(StartPipe), null); | |||||
} | } | ||||
else | else | ||||
{ | { | ||||
@@ -285,15 +285,15 @@ namespace Shadowsocks.Controller | |||||
} | } | ||||
private void startPipe(IAsyncResult ar) | |||||
private void StartPipe(IAsyncResult ar) | |||||
{ | { | ||||
try | try | ||||
{ | { | ||||
connection.EndReceive(ar); | connection.EndReceive(ar); | ||||
remote.BeginReceive(remoteRecvBuffer, 0, RecvSize, 0, | remote.BeginReceive(remoteRecvBuffer, 0, RecvSize, 0, | ||||
new AsyncCallback(pipeRemoteReceiveCallback), null); | |||||
new AsyncCallback(PipeRemoteReceiveCallback), null); | |||||
connection.BeginReceive(connetionRecvBuffer, 0, RecvSize, 0, | connection.BeginReceive(connetionRecvBuffer, 0, RecvSize, 0, | ||||
new AsyncCallback(pipeConnectionReceiveCallback), null); | |||||
new AsyncCallback(PipeConnectionReceiveCallback), null); | |||||
} | } | ||||
catch (Exception e) | catch (Exception e) | ||||
{ | { | ||||
@@ -302,7 +302,7 @@ namespace Shadowsocks.Controller | |||||
} | } | ||||
} | } | ||||
private void pipeRemoteReceiveCallback(IAsyncResult ar) | |||||
private void PipeRemoteReceiveCallback(IAsyncResult ar) | |||||
{ | { | ||||
try | try | ||||
@@ -313,7 +313,7 @@ namespace Shadowsocks.Controller | |||||
{ | { | ||||
int bytesToSend; | int bytesToSend; | ||||
encryptor.Decrypt(remoteRecvBuffer, bytesRead, remoteSendBuffer, out bytesToSend); | encryptor.Decrypt(remoteRecvBuffer, bytesRead, remoteSendBuffer, out bytesToSend); | ||||
connection.BeginSend(remoteSendBuffer, 0, bytesToSend, 0, new AsyncCallback(pipeConnectionSendCallback), null); | |||||
connection.BeginSend(remoteSendBuffer, 0, bytesToSend, 0, new AsyncCallback(PipeConnectionSendCallback), null); | |||||
} | } | ||||
else | else | ||||
{ | { | ||||
@@ -328,7 +328,7 @@ namespace Shadowsocks.Controller | |||||
} | } | ||||
} | } | ||||
private void pipeConnectionReceiveCallback(IAsyncResult ar) | |||||
private void PipeConnectionReceiveCallback(IAsyncResult ar) | |||||
{ | { | ||||
try | try | ||||
@@ -339,7 +339,7 @@ namespace Shadowsocks.Controller | |||||
{ | { | ||||
int bytesToSend; | int bytesToSend; | ||||
encryptor.Encrypt(connetionRecvBuffer, bytesRead, connetionSendBuffer, out bytesToSend); | encryptor.Encrypt(connetionRecvBuffer, bytesRead, connetionSendBuffer, out bytesToSend); | ||||
remote.BeginSend(connetionSendBuffer, 0, bytesToSend, 0, new AsyncCallback(pipeRemoteSendCallback), null); | |||||
remote.BeginSend(connetionSendBuffer, 0, bytesToSend, 0, new AsyncCallback(PipeRemoteSendCallback), null); | |||||
} | } | ||||
else | else | ||||
{ | { | ||||
@@ -353,13 +353,13 @@ namespace Shadowsocks.Controller | |||||
} | } | ||||
} | } | ||||
private void pipeRemoteSendCallback(IAsyncResult ar) | |||||
private void PipeRemoteSendCallback(IAsyncResult ar) | |||||
{ | { | ||||
try | try | ||||
{ | { | ||||
remote.EndSend(ar); | remote.EndSend(ar); | ||||
connection.BeginReceive(this.connetionRecvBuffer, 0, RecvSize, 0, | connection.BeginReceive(this.connetionRecvBuffer, 0, RecvSize, 0, | ||||
new AsyncCallback(pipeConnectionReceiveCallback), null); | |||||
new AsyncCallback(PipeConnectionReceiveCallback), null); | |||||
} | } | ||||
catch (Exception e) | catch (Exception e) | ||||
{ | { | ||||
@@ -368,13 +368,13 @@ namespace Shadowsocks.Controller | |||||
} | } | ||||
} | } | ||||
private void pipeConnectionSendCallback(IAsyncResult ar) | |||||
private void PipeConnectionSendCallback(IAsyncResult ar) | |||||
{ | { | ||||
try | try | ||||
{ | { | ||||
connection.EndSend(ar); | connection.EndSend(ar); | ||||
remote.BeginReceive(this.remoteRecvBuffer, 0, RecvSize, 0, | remote.BeginReceive(this.remoteRecvBuffer, 0, RecvSize, 0, | ||||
new AsyncCallback(pipeRemoteReceiveCallback), null); | |||||
new AsyncCallback(PipeRemoteReceiveCallback), null); | |||||
} | } | ||||
catch (Exception e) | catch (Exception e) | ||||
{ | { | ||||
@@ -1,4 +1,5 @@ | |||||
using Shadowsocks.Model; | |||||
using System.IO; | |||||
using Shadowsocks.Model; | |||||
using System; | using System; | ||||
using System.Collections.Generic; | using System.Collections.Generic; | ||||
using System.Text; | using System.Text; | ||||
@@ -34,6 +35,7 @@ namespace Shadowsocks.Controller | |||||
public ShadowsocksController() | public ShadowsocksController() | ||||
{ | { | ||||
_config = Configuration.Load(); | _config = Configuration.Load(); | ||||
polipoRunner = new PolipoRunner(); | polipoRunner = new PolipoRunner(); | ||||
polipoRunner.Start(_config); | polipoRunner.Start(_config); | ||||
local = new Local(_config); | local = new Local(_config); | ||||
@@ -63,9 +65,10 @@ namespace Shadowsocks.Controller | |||||
return Configuration.Load(); | return Configuration.Load(); | ||||
} | } | ||||
public void SaveServers(List<Server> servers) | |||||
public void SaveServers(List<Server> servers, bool noChange) | |||||
{ | { | ||||
_config.configs = servers; | _config.configs = servers; | ||||
_config.noChange = noChange; | |||||
SaveConfig(_config); | SaveConfig(_config); | ||||
} | } | ||||
@@ -83,6 +86,7 @@ namespace Shadowsocks.Controller | |||||
public void ToggleShareOverLAN(bool enabled) | public void ToggleShareOverLAN(bool enabled) | ||||
{ | { | ||||
_config.shareOverLan = enabled; | _config.shareOverLan = enabled; | ||||
_config.noChange = false; | |||||
SaveConfig(_config); | SaveConfig(_config); | ||||
if (ShareOverLANStatusChanged != null) | if (ShareOverLANStatusChanged != null) | ||||
{ | { | ||||
@@ -129,9 +133,13 @@ namespace Shadowsocks.Controller | |||||
} | } | ||||
protected void SaveConfig(Configuration newConfig) | |||||
public void SaveConfig(Configuration newConfig) | |||||
{ | { | ||||
Configuration.Save(newConfig); | Configuration.Save(newConfig); | ||||
if (newConfig.noChange) | |||||
{ | |||||
return; | |||||
} | |||||
// some logic in configuration updated the config when saving, we need to read it again | // some logic in configuration updated the config when saving, we need to read it again | ||||
_config = Configuration.Load(); | _config = Configuration.Load(); | ||||
@@ -172,6 +180,5 @@ namespace Shadowsocks.Controller | |||||
{ | { | ||||
UpdateSystemProxy(); | UpdateSystemProxy(); | ||||
} | } | ||||
} | } | ||||
} | } |
@@ -17,6 +17,7 @@ namespace Shadowsocks.Model | |||||
public bool enabled; | public bool enabled; | ||||
public bool shareOverLan; | public bool shareOverLan; | ||||
public bool isDefault; | public bool isDefault; | ||||
public bool noChange; | |||||
private static string CONFIG_FILE = "gui-config.json"; | private static string CONFIG_FILE = "gui-config.json"; | ||||
@@ -54,7 +54,6 @@ namespace Shadowsocks | |||||
// TODO run without a main form to save RAM | // TODO run without a main form to save RAM | ||||
Application.Run(new ConfigForm(controller)); | Application.Run(new ConfigForm(controller)); | ||||
} | } | ||||
} | } | ||||
} | } | ||||
@@ -0,0 +1,105 @@ | |||||
namespace Shadowsocks.View | |||||
{ | |||||
partial class AboutForm | |||||
{ | |||||
/// <summary> | |||||
/// Required designer variable. | |||||
/// </summary> | |||||
private System.ComponentModel.IContainer components = null; | |||||
/// <summary> | |||||
/// Clean up any resources being used. | |||||
/// </summary> | |||||
/// <param name="disposing">true if managed resources should be disposed; otherwise, false.</param> | |||||
protected override void Dispose(bool disposing) | |||||
{ | |||||
if (disposing && (components != null)) | |||||
{ | |||||
components.Dispose(); | |||||
} | |||||
base.Dispose(disposing); | |||||
} | |||||
#region Windows Form Designer generated code | |||||
/// <summary> | |||||
/// Required method for Designer support - do not modify | |||||
/// the contents of this method with the code editor. | |||||
/// </summary> | |||||
private void InitializeComponent() | |||||
{ | |||||
this.titleLabel = new System.Windows.Forms.Label(); | |||||
this.versionLabel = new System.Windows.Forms.Label(); | |||||
this.authorLabel = new System.Windows.Forms.Label(); | |||||
this.githubLabel = new System.Windows.Forms.LinkLabel(); | |||||
this.SuspendLayout(); | |||||
// | |||||
// titleLabel | |||||
// | |||||
this.titleLabel.Font = new System.Drawing.Font("Microsoft Sans Serif", 14.25F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0))); | |||||
this.titleLabel.Location = new System.Drawing.Point(12, 40); | |||||
this.titleLabel.Name = "titleLabel"; | |||||
this.titleLabel.Size = new System.Drawing.Size(260, 23); | |||||
this.titleLabel.TabIndex = 0; | |||||
this.titleLabel.Text = "Shadowsocks for Windows"; | |||||
this.titleLabel.TextAlign = System.Drawing.ContentAlignment.MiddleCenter; | |||||
// | |||||
// versionLabel | |||||
// | |||||
this.versionLabel.Font = new System.Drawing.Font("Microsoft Sans Serif", 12F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0))); | |||||
this.versionLabel.Location = new System.Drawing.Point(12, 84); | |||||
this.versionLabel.Name = "versionLabel"; | |||||
this.versionLabel.Size = new System.Drawing.Size(260, 23); | |||||
this.versionLabel.TabIndex = 0; | |||||
this.versionLabel.Text = "Version: 2.0.4.0"; | |||||
this.versionLabel.TextAlign = System.Drawing.ContentAlignment.MiddleCenter; | |||||
// | |||||
// authorLabel | |||||
// | |||||
this.authorLabel.Font = new System.Drawing.Font("Microsoft Sans Serif", 12F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0))); | |||||
this.authorLabel.Location = new System.Drawing.Point(12, 139); | |||||
this.authorLabel.Name = "authorLabel"; | |||||
this.authorLabel.Size = new System.Drawing.Size(260, 23); | |||||
this.authorLabel.TabIndex = 0; | |||||
this.authorLabel.Text = "By: clowwindy"; | |||||
this.authorLabel.TextAlign = System.Drawing.ContentAlignment.MiddleCenter; | |||||
// | |||||
// githubLabel | |||||
// | |||||
this.githubLabel.Font = new System.Drawing.Font("Microsoft Sans Serif", 14.25F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((byte)(0))); | |||||
this.githubLabel.Location = new System.Drawing.Point(13, 201); | |||||
this.githubLabel.Name = "githubLabel"; | |||||
this.githubLabel.Size = new System.Drawing.Size(259, 23); | |||||
this.githubLabel.TabIndex = 1; | |||||
this.githubLabel.TabStop = true; | |||||
this.githubLabel.Text = "GitHub"; | |||||
this.githubLabel.TextAlign = System.Drawing.ContentAlignment.MiddleCenter; | |||||
this.githubLabel.LinkClicked += new System.Windows.Forms.LinkLabelLinkClickedEventHandler(this.githubLabel_LinkClicked); | |||||
// | |||||
// AboutForm | |||||
// | |||||
this.AutoScaleDimensions = new System.Drawing.SizeF(6F, 13F); | |||||
this.AutoScaleMode = System.Windows.Forms.AutoScaleMode.Font; | |||||
this.ClientSize = new System.Drawing.Size(284, 261); | |||||
this.Controls.Add(this.githubLabel); | |||||
this.Controls.Add(this.authorLabel); | |||||
this.Controls.Add(this.versionLabel); | |||||
this.Controls.Add(this.titleLabel); | |||||
this.MaximizeBox = false; | |||||
this.MinimizeBox = false; | |||||
this.Name = "AboutForm"; | |||||
this.StartPosition = System.Windows.Forms.FormStartPosition.CenterScreen; | |||||
this.Text = "About"; | |||||
this.Load += new System.EventHandler(this.AboutForm_Load); | |||||
this.ResumeLayout(false); | |||||
} | |||||
#endregion | |||||
private System.Windows.Forms.Label titleLabel; | |||||
private System.Windows.Forms.Label versionLabel; | |||||
private System.Windows.Forms.Label authorLabel; | |||||
private System.Windows.Forms.LinkLabel githubLabel; | |||||
} | |||||
} |
@@ -0,0 +1,34 @@ | |||||
using System; | |||||
using System.Collections.Generic; | |||||
using System.ComponentModel; | |||||
using System.Data; | |||||
using System.Diagnostics; | |||||
using System.Drawing; | |||||
using System.Reflection; | |||||
using System.Text; | |||||
using System.Windows.Forms; | |||||
namespace Shadowsocks.View | |||||
{ | |||||
public partial class AboutForm : Form | |||||
{ | |||||
public AboutForm() | |||||
{ | |||||
InitializeComponent(); | |||||
} | |||||
private void githubLabel_LinkClicked(object sender, LinkLabelLinkClickedEventArgs e) | |||||
{ | |||||
string url = "https://github.com/clowwindy/shadowsocks-csharp"; | |||||
System.Diagnostics.Process.Start(url); | |||||
} | |||||
private void AboutForm_Load(object sender, EventArgs e) | |||||
{ | |||||
Assembly assembly = Assembly.GetExecutingAssembly(); | |||||
FileVersionInfo fvi = FileVersionInfo.GetVersionInfo(assembly.Location); | |||||
string version = fvi.FileVersion; | |||||
versionLabel.Text = "Version: " + version; | |||||
} | |||||
} | |||||
} |
@@ -0,0 +1,120 @@ | |||||
<?xml version="1.0" encoding="utf-8"?> | |||||
<root> | |||||
<!-- | |||||
Microsoft ResX Schema | |||||
Version 2.0 | |||||
The primary goals of this format is to allow a simple XML format | |||||
that is mostly human readable. The generation and parsing of the | |||||
various data types are done through the TypeConverter classes | |||||
associated with the data types. | |||||
Example: | |||||
... ado.net/XML headers & schema ... | |||||
<resheader name="resmimetype">text/microsoft-resx</resheader> | |||||
<resheader name="version">2.0</resheader> | |||||
<resheader name="reader">System.Resources.ResXResourceReader, System.Windows.Forms, ...</resheader> | |||||
<resheader name="writer">System.Resources.ResXResourceWriter, System.Windows.Forms, ...</resheader> | |||||
<data name="Name1"><value>this is my long string</value><comment>this is a comment</comment></data> | |||||
<data name="Color1" type="System.Drawing.Color, System.Drawing">Blue</data> | |||||
<data name="Bitmap1" mimetype="application/x-microsoft.net.object.binary.base64"> | |||||
<value>[base64 mime encoded serialized .NET Framework object]</value> | |||||
</data> | |||||
<data name="Icon1" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64"> | |||||
<value>[base64 mime encoded string representing a byte array form of the .NET Framework object]</value> | |||||
<comment>This is a comment</comment> | |||||
</data> | |||||
There are any number of "resheader" rows that contain simple | |||||
name/value pairs. | |||||
Each data row contains a name, and value. The row also contains a | |||||
type or mimetype. Type corresponds to a .NET class that support | |||||
text/value conversion through the TypeConverter architecture. | |||||
Classes that don't support this are serialized and stored with the | |||||
mimetype set. | |||||
The mimetype is used for serialized objects, and tells the | |||||
ResXResourceReader how to depersist the object. This is currently not | |||||
extensible. For a given mimetype the value must be set accordingly: | |||||
Note - application/x-microsoft.net.object.binary.base64 is the format | |||||
that the ResXResourceWriter will generate, however the reader can | |||||
read any of the formats listed below. | |||||
mimetype: application/x-microsoft.net.object.binary.base64 | |||||
value : The object must be serialized with | |||||
: System.Runtime.Serialization.Formatters.Binary.BinaryFormatter | |||||
: and then encoded with base64 encoding. | |||||
mimetype: application/x-microsoft.net.object.soap.base64 | |||||
value : The object must be serialized with | |||||
: System.Runtime.Serialization.Formatters.Soap.SoapFormatter | |||||
: and then encoded with base64 encoding. | |||||
mimetype: application/x-microsoft.net.object.bytearray.base64 | |||||
value : The object must be serialized into a byte array | |||||
: using a System.ComponentModel.TypeConverter | |||||
: and then encoded with base64 encoding. | |||||
--> | |||||
<xsd:schema id="root" xmlns="" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata"> | |||||
<xsd:import namespace="http://www.w3.org/XML/1998/namespace" /> | |||||
<xsd:element name="root" msdata:IsDataSet="true"> | |||||
<xsd:complexType> | |||||
<xsd:choice maxOccurs="unbounded"> | |||||
<xsd:element name="metadata"> | |||||
<xsd:complexType> | |||||
<xsd:sequence> | |||||
<xsd:element name="value" type="xsd:string" minOccurs="0" /> | |||||
</xsd:sequence> | |||||
<xsd:attribute name="name" use="required" type="xsd:string" /> | |||||
<xsd:attribute name="type" type="xsd:string" /> | |||||
<xsd:attribute name="mimetype" type="xsd:string" /> | |||||
<xsd:attribute ref="xml:space" /> | |||||
</xsd:complexType> | |||||
</xsd:element> | |||||
<xsd:element name="assembly"> | |||||
<xsd:complexType> | |||||
<xsd:attribute name="alias" type="xsd:string" /> | |||||
<xsd:attribute name="name" type="xsd:string" /> | |||||
</xsd:complexType> | |||||
</xsd:element> | |||||
<xsd:element name="data"> | |||||
<xsd:complexType> | |||||
<xsd:sequence> | |||||
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" /> | |||||
<xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" /> | |||||
</xsd:sequence> | |||||
<xsd:attribute name="name" type="xsd:string" use="required" msdata:Ordinal="1" /> | |||||
<xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" /> | |||||
<xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" /> | |||||
<xsd:attribute ref="xml:space" /> | |||||
</xsd:complexType> | |||||
</xsd:element> | |||||
<xsd:element name="resheader"> | |||||
<xsd:complexType> | |||||
<xsd:sequence> | |||||
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" /> | |||||
</xsd:sequence> | |||||
<xsd:attribute name="name" type="xsd:string" use="required" /> | |||||
</xsd:complexType> | |||||
</xsd:element> | |||||
</xsd:choice> | |||||
</xsd:complexType> | |||||
</xsd:element> | |||||
</xsd:schema> | |||||
<resheader name="resmimetype"> | |||||
<value>text/microsoft-resx</value> | |||||
</resheader> | |||||
<resheader name="version"> | |||||
<value>2.0</value> | |||||
</resheader> | |||||
<resheader name="reader"> | |||||
<value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> | |||||
</resheader> | |||||
<resheader name="writer"> | |||||
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value> | |||||
</resheader> | |||||
</root> |
@@ -14,11 +14,13 @@ namespace Shadowsocks.View | |||||
{ | { | ||||
private ShadowsocksController controller; | private ShadowsocksController controller; | ||||
private UpdateChecker updateChecker; | private UpdateChecker updateChecker; | ||||
private AboutForm aboutForm; | |||||
// this is a copy of configuration that we are working on | // this is a copy of configuration that we are working on | ||||
private Configuration modifiedConfiguration; | |||||
private int oldSelectedIndex = -1; | |||||
private bool isFirstRun; | |||||
private Configuration _modifiedConfiguration; | |||||
private Configuration _oldConfiguration; | |||||
private int _oldSelectedIndex = -1; | |||||
private bool _isFirstRun; | |||||
public ConfigForm(ShadowsocksController controller) | public ConfigForm(ShadowsocksController controller) | ||||
{ | { | ||||
@@ -66,7 +68,7 @@ namespace Shadowsocks.View | |||||
notifyIcon1.BalloonTipIcon = ToolTipIcon.Info; | notifyIcon1.BalloonTipIcon = ToolTipIcon.Info; | ||||
notifyIcon1.BalloonTipClicked += notifyIcon1_BalloonTipClicked; | notifyIcon1.BalloonTipClicked += notifyIcon1_BalloonTipClicked; | ||||
notifyIcon1.ShowBalloonTip(5000); | notifyIcon1.ShowBalloonTip(5000); | ||||
isFirstRun = false; | |||||
_isFirstRun = false; | |||||
} | } | ||||
void notifyIcon1_BalloonTipClicked(object sender, EventArgs e) | void notifyIcon1_BalloonTipClicked(object sender, EventArgs e) | ||||
@@ -82,11 +84,11 @@ namespace Shadowsocks.View | |||||
IPTextBox.Focus(); | IPTextBox.Focus(); | ||||
} | } | ||||
private bool SaveOldSelectedServer() | |||||
private bool SaveOldSelectedServer(object sender) | |||||
{ | { | ||||
try | try | ||||
{ | { | ||||
if (oldSelectedIndex == -1 || oldSelectedIndex >= modifiedConfiguration.configs.Count) | |||||
if (_oldSelectedIndex == -1 || _oldSelectedIndex >= _modifiedConfiguration.configs.Count) | |||||
{ | { | ||||
return true; | return true; | ||||
} | } | ||||
@@ -100,7 +102,27 @@ namespace Shadowsocks.View | |||||
remarks = RemarksTextBox.Text | remarks = RemarksTextBox.Text | ||||
}; | }; | ||||
Configuration.CheckServer(server); | Configuration.CheckServer(server); | ||||
modifiedConfiguration.configs[oldSelectedIndex] = server; | |||||
_modifiedConfiguration.configs[_oldSelectedIndex] = server; | |||||
if (sender.Equals(OKButton)) | |||||
{ | |||||
if (_oldConfiguration.configs[_oldConfiguration.index].server == | |||||
_modifiedConfiguration.configs[_oldSelectedIndex].server && | |||||
_oldConfiguration.configs[_oldConfiguration.index].server_port == | |||||
_modifiedConfiguration.configs[_oldSelectedIndex].server_port && | |||||
_oldConfiguration.configs[_oldConfiguration.index].password == | |||||
_modifiedConfiguration.configs[_oldSelectedIndex].password && | |||||
_oldConfiguration.configs[_oldConfiguration.index].local_port == | |||||
_modifiedConfiguration.configs[_oldSelectedIndex].local_port) | |||||
{ | |||||
_modifiedConfiguration.noChange = true; | |||||
} | |||||
else | |||||
{ | |||||
_modifiedConfiguration.noChange = false; | |||||
_oldConfiguration = _modifiedConfiguration; | |||||
} | |||||
} | |||||
return true; | return true; | ||||
} | } | ||||
catch (FormatException) | catch (FormatException) | ||||
@@ -116,15 +138,15 @@ namespace Shadowsocks.View | |||||
private void LoadSelectedServer() | private void LoadSelectedServer() | ||||
{ | { | ||||
if (ServersListBox.SelectedIndex >= 0 && ServersListBox.SelectedIndex < modifiedConfiguration.configs.Count) | |||||
if (ServersListBox.SelectedIndex >= 0 && ServersListBox.SelectedIndex < _modifiedConfiguration.configs.Count) | |||||
{ | { | ||||
Server server = modifiedConfiguration.configs[ServersListBox.SelectedIndex]; | |||||
Server server = _modifiedConfiguration.configs[ServersListBox.SelectedIndex]; | |||||
IPTextBox.Text = server.server; | IPTextBox.Text = server.server; | ||||
ServerPortTextBox.Text = server.server_port.ToString(); | ServerPortTextBox.Text = server.server_port.ToString(); | ||||
PasswordTextBox.Text = server.password; | PasswordTextBox.Text = server.password; | ||||
ProxyPortTextBox.Text = server.local_port.ToString(); | ProxyPortTextBox.Text = server.local_port.ToString(); | ||||
EncryptionSelect.Text = server.method == null ? "aes-256-cfb" : server.method; | |||||
EncryptionSelect.Text = server.method ?? "aes-256-cfb"; | |||||
RemarksTextBox.Text = server.remarks; | RemarksTextBox.Text = server.remarks; | ||||
ServerGroupBox.Visible = true; | ServerGroupBox.Visible = true; | ||||
//IPTextBox.Focus(); | //IPTextBox.Focus(); | ||||
@@ -138,7 +160,7 @@ namespace Shadowsocks.View | |||||
private void LoadConfiguration(Configuration configuration) | private void LoadConfiguration(Configuration configuration) | ||||
{ | { | ||||
ServersListBox.Items.Clear(); | ServersListBox.Items.Clear(); | ||||
foreach (Server server in modifiedConfiguration.configs) | |||||
foreach (Server server in _modifiedConfiguration.configs) | |||||
{ | { | ||||
ServersListBox.Items.Add(string.IsNullOrEmpty(server.server) ? "New server" : string.IsNullOrEmpty(server.remarks)? server.server + ":" + server.server_port : server.server + ":" + server.server_port + " (" + server.remarks + ")"); | ServersListBox.Items.Add(string.IsNullOrEmpty(server.server) ? "New server" : string.IsNullOrEmpty(server.remarks)? server.server + ":" + server.server_port : server.server + ":" + server.server_port + " (" + server.remarks + ")"); | ||||
} | } | ||||
@@ -146,15 +168,16 @@ namespace Shadowsocks.View | |||||
private void LoadCurrentConfiguration() | private void LoadCurrentConfiguration() | ||||
{ | { | ||||
modifiedConfiguration = controller.GetConfiguration(); | |||||
LoadConfiguration(modifiedConfiguration); | |||||
oldSelectedIndex = modifiedConfiguration.index; | |||||
ServersListBox.SelectedIndex = modifiedConfiguration.index; | |||||
_modifiedConfiguration = controller.GetConfiguration(); | |||||
_oldConfiguration = _modifiedConfiguration; | |||||
LoadConfiguration(_modifiedConfiguration); | |||||
_oldSelectedIndex = _modifiedConfiguration.index; | |||||
ServersListBox.SelectedIndex = _modifiedConfiguration.index; | |||||
LoadSelectedServer(); | LoadSelectedServer(); | ||||
UpdateServersMenu(); | UpdateServersMenu(); | ||||
enableItem.Checked = modifiedConfiguration.enabled; | |||||
ShareOverLANItem.Checked = modifiedConfiguration.shareOverLan; | |||||
enableItem.Checked = _modifiedConfiguration.enabled; | |||||
ShareOverLANItem.Checked = _modifiedConfiguration.shareOverLan; | |||||
} | } | ||||
private void UpdateServersMenu() | private void UpdateServersMenu() | ||||
@@ -193,56 +216,56 @@ namespace Shadowsocks.View | |||||
} | } | ||||
else | else | ||||
{ | { | ||||
isFirstRun = true; | |||||
_isFirstRun = true; | |||||
} | } | ||||
updateChecker.CheckUpdate(); | updateChecker.CheckUpdate(); | ||||
} | } | ||||
private void ServersListBox_SelectedIndexChanged(object sender, EventArgs e) | private void ServersListBox_SelectedIndexChanged(object sender, EventArgs e) | ||||
{ | { | ||||
if (oldSelectedIndex == ServersListBox.SelectedIndex) | |||||
if (_oldSelectedIndex == ServersListBox.SelectedIndex) | |||||
{ | { | ||||
// we are moving back to oldSelectedIndex or doing a force move | // we are moving back to oldSelectedIndex or doing a force move | ||||
return; | return; | ||||
} | } | ||||
if (!SaveOldSelectedServer()) | |||||
if (!SaveOldSelectedServer(sender)) | |||||
{ | { | ||||
// why this won't cause stack overflow? | // why this won't cause stack overflow? | ||||
ServersListBox.SelectedIndex = oldSelectedIndex; | |||||
ServersListBox.SelectedIndex = _oldSelectedIndex; | |||||
return; | return; | ||||
} | } | ||||
LoadSelectedServer(); | LoadSelectedServer(); | ||||
oldSelectedIndex = ServersListBox.SelectedIndex; | |||||
_oldSelectedIndex = ServersListBox.SelectedIndex; | |||||
} | } | ||||
private void AddButton_Click(object sender, EventArgs e) | private void AddButton_Click(object sender, EventArgs e) | ||||
{ | { | ||||
if (!SaveOldSelectedServer()) | |||||
if (!SaveOldSelectedServer(sender)) | |||||
{ | { | ||||
return; | return; | ||||
} | } | ||||
Server server = Configuration.GetDefaultServer(); | Server server = Configuration.GetDefaultServer(); | ||||
modifiedConfiguration.configs.Add(server); | |||||
LoadConfiguration(modifiedConfiguration); | |||||
ServersListBox.SelectedIndex = modifiedConfiguration.configs.Count - 1; | |||||
oldSelectedIndex = ServersListBox.SelectedIndex; | |||||
_modifiedConfiguration.configs.Add(server); | |||||
LoadConfiguration(_modifiedConfiguration); | |||||
ServersListBox.SelectedIndex = _modifiedConfiguration.configs.Count - 1; | |||||
_oldSelectedIndex = ServersListBox.SelectedIndex; | |||||
} | } | ||||
private void DeleteButton_Click(object sender, EventArgs e) | private void DeleteButton_Click(object sender, EventArgs e) | ||||
{ | { | ||||
oldSelectedIndex = ServersListBox.SelectedIndex; | |||||
if (oldSelectedIndex >= 0 && oldSelectedIndex < modifiedConfiguration.configs.Count) | |||||
_oldSelectedIndex = ServersListBox.SelectedIndex; | |||||
if (_oldSelectedIndex >= 0 && _oldSelectedIndex < _modifiedConfiguration.configs.Count) | |||||
{ | { | ||||
modifiedConfiguration.configs.RemoveAt(oldSelectedIndex); | |||||
_modifiedConfiguration.configs.RemoveAt(_oldSelectedIndex); | |||||
} | } | ||||
if (oldSelectedIndex >= modifiedConfiguration.configs.Count) | |||||
if (_oldSelectedIndex >= _modifiedConfiguration.configs.Count) | |||||
{ | { | ||||
// can be -1 | // can be -1 | ||||
oldSelectedIndex = modifiedConfiguration.configs.Count - 1; | |||||
_oldSelectedIndex = _modifiedConfiguration.configs.Count - 1; | |||||
} | } | ||||
ServersListBox.SelectedIndex = oldSelectedIndex; | |||||
LoadConfiguration(modifiedConfiguration); | |||||
ServersListBox.SelectedIndex = oldSelectedIndex; | |||||
ServersListBox.SelectedIndex = _oldSelectedIndex; | |||||
LoadConfiguration(_modifiedConfiguration); | |||||
ServersListBox.SelectedIndex = _oldSelectedIndex; | |||||
LoadSelectedServer(); | LoadSelectedServer(); | ||||
} | } | ||||
@@ -258,28 +281,28 @@ namespace Shadowsocks.View | |||||
private void ShowFirstTimeBalloon() | private void ShowFirstTimeBalloon() | ||||
{ | { | ||||
if (isFirstRun) | |||||
if (_isFirstRun) | |||||
{ | { | ||||
notifyIcon1.BalloonTipTitle = "Shadowsocks is here"; | notifyIcon1.BalloonTipTitle = "Shadowsocks is here"; | ||||
notifyIcon1.BalloonTipText = "You can turn on/off Shadowsocks in the context menu"; | notifyIcon1.BalloonTipText = "You can turn on/off Shadowsocks in the context menu"; | ||||
notifyIcon1.BalloonTipIcon = ToolTipIcon.Info; | notifyIcon1.BalloonTipIcon = ToolTipIcon.Info; | ||||
notifyIcon1.ShowBalloonTip(0); | notifyIcon1.ShowBalloonTip(0); | ||||
isFirstRun = false; | |||||
_isFirstRun = false; | |||||
} | } | ||||
} | } | ||||
private void OKButton_Click(object sender, EventArgs e) | private void OKButton_Click(object sender, EventArgs e) | ||||
{ | { | ||||
if (!SaveOldSelectedServer()) | |||||
if (!SaveOldSelectedServer(sender)) | |||||
{ | { | ||||
return; | return; | ||||
} | } | ||||
if (modifiedConfiguration.configs.Count == 0) | |||||
if (_modifiedConfiguration.configs.Count == 0) | |||||
{ | { | ||||
MessageBox.Show("Please add at least one server"); | MessageBox.Show("Please add at least one server"); | ||||
return; | return; | ||||
} | } | ||||
controller.SaveServers(modifiedConfiguration.configs); | |||||
controller.SaveServers(_modifiedConfiguration.configs, _modifiedConfiguration.noChange); | |||||
this.Hide(); | this.Hide(); | ||||
ShowFirstTimeBalloon(); | ShowFirstTimeBalloon(); | ||||
} | } | ||||
@@ -298,7 +321,9 @@ namespace Shadowsocks.View | |||||
private void AboutItem_Click(object sender, EventArgs e) | private void AboutItem_Click(object sender, EventArgs e) | ||||
{ | { | ||||
Process.Start("https://github.com/clowwindy/shadowsocks-csharp"); | |||||
aboutForm = new AboutForm(); | |||||
aboutForm.Show(); | |||||
//Process.Start("https://github.com/clowwindy/shadowsocks-csharp"); | |||||
} | } | ||||
private void notifyIcon1_DoubleClick(object sender, EventArgs e) | private void notifyIcon1_DoubleClick(object sender, EventArgs e) | ||||
@@ -348,5 +373,13 @@ namespace Shadowsocks.View | |||||
qrCodeForm.Icon = this.Icon; | qrCodeForm.Icon = this.Icon; | ||||
qrCodeForm.Show(); | qrCodeForm.Show(); | ||||
} | } | ||||
private void enableLogBox_CheckedChanged(object sender, EventArgs e) | |||||
{ | |||||
if (this.Visible) | |||||
{ | |||||
MessageBox.Show("This option only works on next startup."); | |||||
} | |||||
} | |||||
} | } | ||||
} | } |
@@ -59,11 +59,11 @@ | |||||
<Prefer32Bit>false</Prefer32Bit> | <Prefer32Bit>false</Prefer32Bit> | ||||
</PropertyGroup> | </PropertyGroup> | ||||
<ItemGroup> | <ItemGroup> | ||||
<Reference Include="System" /> | |||||
<Reference Include="System" /> | |||||
<Reference Include="System.Data" /> | <Reference Include="System.Data" /> | ||||
<Reference Include="System.Drawing" /> | |||||
<Reference Include="System.Windows.Forms" /> | |||||
<Reference Include="System.XML" /> | |||||
<Reference Include="System.Drawing" /> | |||||
<Reference Include="System.Windows.Forms" /> | |||||
<Reference Include="System.XML" /> | |||||
</ItemGroup> | </ItemGroup> | ||||
<ItemGroup> | <ItemGroup> | ||||
<Compile Include="3rd\QRCodeCS.cs" /> | <Compile Include="3rd\QRCodeCS.cs" /> | ||||
@@ -80,6 +80,12 @@ | |||||
<Compile Include="Controller\PACServer.cs" /> | <Compile Include="Controller\PACServer.cs" /> | ||||
<Compile Include="Model\Server.cs" /> | <Compile Include="Model\Server.cs" /> | ||||
<Compile Include="Model\Configuration.cs" /> | <Compile Include="Model\Configuration.cs" /> | ||||
<Compile Include="View\AboutForm.cs"> | |||||
<SubType>Form</SubType> | |||||
</Compile> | |||||
<Compile Include="View\AboutForm.Designer.cs"> | |||||
<DependentUpon>AboutForm.cs</DependentUpon> | |||||
</Compile> | |||||
<Compile Include="View\ConfigForm.cs"> | <Compile Include="View\ConfigForm.cs"> | ||||
<SubType>Form</SubType> | <SubType>Form</SubType> | ||||
</Compile> | </Compile> | ||||
@@ -98,6 +104,9 @@ | |||||
<Compile Include="View\QRCodeForm.Designer.cs"> | <Compile Include="View\QRCodeForm.Designer.cs"> | ||||
<DependentUpon>QRCodeForm.cs</DependentUpon> | <DependentUpon>QRCodeForm.cs</DependentUpon> | ||||
</Compile> | </Compile> | ||||
<EmbeddedResource Include="View\AboutForm.resx"> | |||||
<DependentUpon>AboutForm.cs</DependentUpon> | |||||
</EmbeddedResource> | |||||
<EmbeddedResource Include="View\ConfigForm.resx"> | <EmbeddedResource Include="View\ConfigForm.resx"> | ||||
<DependentUpon>ConfigForm.cs</DependentUpon> | <DependentUpon>ConfigForm.cs</DependentUpon> | ||||
<SubType>Designer</SubType> | <SubType>Designer</SubType> | ||||