Compare commits
78 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
790209efbc | ||
|
|
f1679e444c | ||
|
|
9dc6ba182a | ||
|
|
785a30e623 | ||
|
|
f09efdad66 | ||
|
|
4e73b3ae28 | ||
|
|
6213f86f81 | ||
|
|
0a6955cd59 | ||
|
|
6ee8f03ec0 | ||
|
|
b1a82d95c2 | ||
|
|
30bc9ded29 | ||
|
|
5a32892e94 | ||
|
|
e33de896b6 | ||
|
|
3ba92444a5 | ||
|
|
4f120e8eb4 | ||
|
|
08aebf5736 | ||
|
|
5a4966ba8d | ||
|
|
9ba963fc45 | ||
|
|
ccc10dbae4 | ||
|
|
d858342269 | ||
|
|
63f251d1fd | ||
|
|
83efe66f3e | ||
|
|
3f5729044f | ||
|
|
045af7e8df | ||
|
|
6f181053b2 | ||
|
|
4ff1dc2982 | ||
|
|
a883ba8808 | ||
|
|
961bd6140c | ||
|
|
5f364b48c9 | ||
|
|
5a04911c7c | ||
|
|
8ff248aa62 | ||
|
|
046ac95dc3 | ||
|
|
6fb17a4b74 | ||
|
|
f84397393d | ||
|
|
800c93e2aa | ||
|
|
df6179a1a8 | ||
|
|
3e3a079ba1 | ||
|
|
0bf4a43663 | ||
|
|
5e9f4ad926 | ||
|
|
69050bfe41 | ||
|
|
eebc16bcdd | ||
|
|
d4921535f2 | ||
|
|
b1eeb648a7 | ||
|
|
ba702ba041 | ||
|
|
c40d88d0b6 | ||
|
|
285f91e9e8 | ||
|
|
947a7aa7df | ||
|
|
0b37e36036 | ||
|
|
92320f5086 | ||
|
|
ae17e0c264 | ||
|
|
ec756ee943 | ||
|
|
b0ff814753 | ||
|
|
1f7eb2d48a | ||
|
|
eeab8e4a90 | ||
|
|
77fbddf488 | ||
|
|
bf396f8802 | ||
|
|
13cb8b84bf | ||
|
|
3ab992f5fb | ||
|
|
a386ecfc9c | ||
|
|
a71399c42c | ||
|
|
5a5d7e0981 | ||
|
|
b663a7f52a | ||
|
|
7390039086 | ||
|
|
12af02435f | ||
|
|
6f357828b7 | ||
|
|
09e1a4386d | ||
|
|
dcc8e6dc34 | ||
|
|
1b19ef54e4 | ||
|
|
8ce476caf1 | ||
|
|
dd2d9133eb | ||
|
|
2147d12ac8 | ||
|
|
22641a1da0 | ||
|
|
2bd4088d40 | ||
|
|
ee61363c31 | ||
|
|
8285688ec9 | ||
|
|
4eb076540e | ||
|
|
527bfec24c | ||
|
|
2fff7d7160 |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -16,3 +16,4 @@
|
|||||||
/v2rayN/v2rayUpgrade/bin/Release
|
/v2rayN/v2rayUpgrade/bin/Release
|
||||||
/v2rayN/v2rayUpgrade/obj/
|
/v2rayN/v2rayUpgrade/obj/
|
||||||
*.user
|
*.user
|
||||||
|
/.vs/v2rayN
|
||||||
|
|||||||
@@ -20,14 +20,8 @@ public class PacHandler
|
|||||||
|
|
||||||
public static void Start(string configPath, int httpPort, int pacPort)
|
public static void Start(string configPath, int httpPort, int pacPort)
|
||||||
{
|
{
|
||||||
if (configPath.Equals(_configPath)
|
_needRestart = (configPath != _configPath || httpPort != _httpPort || pacPort != _pacPort || !_isRunning);
|
||||||
&& httpPort.Equals(_httpPort)
|
|
||||||
&& pacPort.Equals(_pacPort)
|
|
||||||
&& _isRunning)
|
|
||||||
{
|
|
||||||
_needRestart = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
_configPath = configPath;
|
_configPath = configPath;
|
||||||
_httpPort = httpPort;
|
_httpPort = httpPort;
|
||||||
_pacPort = pacPort;
|
_pacPort = pacPort;
|
||||||
|
|||||||
@@ -42,10 +42,11 @@ namespace v2rayN
|
|||||||
Global.processJob = new Job();
|
Global.processJob = new Job();
|
||||||
|
|
||||||
Logging.Setup();
|
Logging.Setup();
|
||||||
|
Init();
|
||||||
|
Logging.LoggingEnabled(_config.guiItem.enableLog);
|
||||||
Utils.SaveLog($"v2rayN start up | {Utils.GetVersion()} | {Utils.GetExePath()}");
|
Utils.SaveLog($"v2rayN start up | {Utils.GetVersion()} | {Utils.GetExePath()}");
|
||||||
Logging.ClearLogs();
|
Logging.ClearLogs();
|
||||||
|
|
||||||
Init();
|
|
||||||
|
|
||||||
Thread.CurrentThread.CurrentUICulture = new(_config.uiItem.currentLanguage);
|
Thread.CurrentThread.CurrentUICulture = new(_config.uiItem.currentLanguage);
|
||||||
|
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ namespace v2rayN.Base
|
|||||||
private static readonly Lazy<DownloaderHelper> _instance = new(() => new());
|
private static readonly Lazy<DownloaderHelper> _instance = new(() => new());
|
||||||
public static DownloaderHelper Instance => _instance.Value;
|
public static DownloaderHelper Instance => _instance.Value;
|
||||||
|
|
||||||
public async Task<string?> DownloadStringAsync(IWebProxy webProxy, string url, string? userAgent, int timeout)
|
public async Task<string?> DownloadStringAsync(IWebProxy? webProxy, string url, string? userAgent, int timeout)
|
||||||
{
|
{
|
||||||
if (string.IsNullOrEmpty(url))
|
if (string.IsNullOrEmpty(url))
|
||||||
{
|
{
|
||||||
@@ -45,7 +45,7 @@ namespace v2rayN.Base
|
|||||||
{
|
{
|
||||||
if (value.Error != null)
|
if (value.Error != null)
|
||||||
{
|
{
|
||||||
throw new Exception(string.Format("{0}", value.Error.Message));
|
throw value.Error;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
using var stream = await downloader.DownloadFileTaskAsync(address: url, cancellationToken: cancellationToken.Token);
|
using var stream = await downloader.DownloadFileTaskAsync(address: url, cancellationToken: cancellationToken.Token);
|
||||||
@@ -115,14 +115,14 @@ namespace v2rayN.Base
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
progress.Report("......");
|
//progress.Report("......");
|
||||||
|
|
||||||
await downloader.DownloadFileTaskAsync(address: url, cancellationToken: cancellationToken.Token);
|
using var stream = await downloader.DownloadFileTaskAsync(address: url, cancellationToken: cancellationToken.Token);
|
||||||
|
|
||||||
downloadOpt = null;
|
downloadOpt = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task DownloadFileAsync(IWebProxy webProxy, string url, string fileName, IProgress<double> progress, int timeout)
|
public async Task DownloadFileAsync(IWebProxy? webProxy, string url, string fileName, IProgress<double> progress, int timeout)
|
||||||
{
|
{
|
||||||
if (string.IsNullOrEmpty(url))
|
if (string.IsNullOrEmpty(url))
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Net.Http;
|
using System.Net.Http;
|
||||||
using System.Net.Http.Headers;
|
using System.Net.Mime;
|
||||||
|
using System.Text;
|
||||||
|
|
||||||
namespace v2rayN.Base
|
namespace v2rayN.Base
|
||||||
{
|
{
|
||||||
@@ -8,86 +9,46 @@ namespace v2rayN.Base
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public class HttpClientHelper
|
public class HttpClientHelper
|
||||||
{
|
{
|
||||||
private static HttpClientHelper httpClientHelper = null;
|
private readonly static Lazy<HttpClientHelper> _instance = new(() =>
|
||||||
private HttpClient httpClient;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// </summary>
|
|
||||||
private HttpClientHelper() { }
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// </summary>
|
|
||||||
/// <returns></returns>
|
|
||||||
public static HttpClientHelper GetInstance()
|
|
||||||
{
|
{
|
||||||
if (httpClientHelper != null)
|
HttpClientHandler handler = new() { UseCookies = false };
|
||||||
{
|
HttpClientHelper helper = new(new HttpClient(handler));
|
||||||
return httpClientHelper;
|
return helper;
|
||||||
}
|
});
|
||||||
else
|
public static HttpClientHelper Instance => _instance.Value;
|
||||||
{
|
private readonly HttpClient httpClient;
|
||||||
HttpClientHelper httpClientHelper = new();
|
|
||||||
|
private HttpClientHelper(HttpClient httpClient) => this.httpClient = httpClient;
|
||||||
|
|
||||||
HttpClientHandler handler = new() { UseCookies = false };
|
|
||||||
httpClientHelper.httpClient = new HttpClient(handler);
|
|
||||||
return httpClientHelper;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
public async Task<string?> GetAsync(string url)
|
public async Task<string?> GetAsync(string url)
|
||||||
{
|
{
|
||||||
if (string.IsNullOrEmpty(url))
|
if (string.IsNullOrEmpty(url)) return null;
|
||||||
{
|
return await httpClient.GetStringAsync(url);
|
||||||
return null;
|
|
||||||
}
|
|
||||||
HttpResponseMessage response = await httpClient.GetAsync(url);
|
|
||||||
|
|
||||||
return await response.Content.ReadAsStringAsync();
|
|
||||||
}
|
}
|
||||||
public async Task<string?> GetAsync(HttpClient client, string url, CancellationToken token)
|
|
||||||
|
public async Task<string?> GetAsync(HttpClient client, string url, CancellationToken token = default)
|
||||||
{
|
{
|
||||||
if (string.IsNullOrEmpty(url))
|
if (string.IsNullOrWhiteSpace(url)) return null;
|
||||||
{
|
return await client.GetStringAsync(url, token);
|
||||||
return null;
|
|
||||||
}
|
|
||||||
HttpResponseMessage response = await client.GetAsync(url, token);
|
|
||||||
if (!response.IsSuccessStatusCode)
|
|
||||||
{
|
|
||||||
throw new Exception(string.Format("{0}", response.StatusCode));
|
|
||||||
}
|
|
||||||
return await response.Content.ReadAsStringAsync(token);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task PutAsync(string url, Dictionary<string, string> headers)
|
public async Task PutAsync(string url, Dictionary<string, string> headers)
|
||||||
{
|
{
|
||||||
var myContent = Utils.ToJson(headers);
|
var jsonContent = Utils.ToJson(headers);
|
||||||
var buffer = System.Text.Encoding.UTF8.GetBytes(myContent);
|
var content = new StringContent(jsonContent, Encoding.UTF8, MediaTypeNames.Application.Json);
|
||||||
var byteContent = new ByteArrayContent(buffer);
|
|
||||||
byteContent.Headers.ContentType = new MediaTypeHeaderValue("application/json");
|
|
||||||
|
|
||||||
var result = await httpClient.PutAsync(url, byteContent);
|
var result = await httpClient.PutAsync(url, content);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task DownloadFileAsync(HttpClient client, string url, string fileName, IProgress<double> progress, CancellationToken token)
|
public static async Task DownloadFileAsync(HttpClient client, string url, string fileName, IProgress<double>? progress, CancellationToken token = default)
|
||||||
{
|
{
|
||||||
if (string.IsNullOrEmpty(url))
|
ArgumentNullException.ThrowIfNull(url);
|
||||||
{
|
ArgumentNullException.ThrowIfNull(fileName);
|
||||||
throw new ArgumentNullException(nameof(url));
|
if (File.Exists(fileName)) File.Delete(fileName);
|
||||||
}
|
|
||||||
if (string.IsNullOrEmpty(fileName))
|
|
||||||
{
|
|
||||||
throw new ArgumentNullException(nameof(fileName));
|
|
||||||
}
|
|
||||||
if (File.Exists(fileName))
|
|
||||||
{
|
|
||||||
File.Delete(fileName);
|
|
||||||
}
|
|
||||||
|
|
||||||
var response = await client.GetAsync(url, HttpCompletionOption.ResponseHeadersRead, token);
|
using var response = await client.GetAsync(url, HttpCompletionOption.ResponseHeadersRead, token);
|
||||||
|
|
||||||
if (!response.IsSuccessStatusCode)
|
if (!response.IsSuccessStatusCode) throw new Exception(response.StatusCode.ToString());
|
||||||
{
|
|
||||||
throw new Exception(string.Format("{0}", response.StatusCode));
|
|
||||||
}
|
|
||||||
|
|
||||||
var total = response.Content.Headers.ContentLength ?? -1L;
|
var total = response.Content.Headers.ContentLength ?? -1L;
|
||||||
var canReportProgress = total != -1 && progress != null;
|
var canReportProgress = total != -1 && progress != null;
|
||||||
@@ -96,48 +57,36 @@ namespace v2rayN.Base
|
|||||||
using var file = File.Create(fileName);
|
using var file = File.Create(fileName);
|
||||||
var totalRead = 0L;
|
var totalRead = 0L;
|
||||||
var buffer = new byte[1024 * 1024];
|
var buffer = new byte[1024 * 1024];
|
||||||
var isMoreToRead = true;
|
|
||||||
var progressPercentage = 0;
|
var progressPercentage = 0;
|
||||||
|
|
||||||
do
|
while (true)
|
||||||
{
|
{
|
||||||
token.ThrowIfCancellationRequested();
|
token.ThrowIfCancellationRequested();
|
||||||
|
|
||||||
var read = await stream.ReadAsync(buffer, token);
|
var read = await stream.ReadAsync(buffer, token);
|
||||||
|
totalRead += read;
|
||||||
|
|
||||||
if (read == 0)
|
if (read == 0) break;
|
||||||
|
file.Write(buffer, 0, read);
|
||||||
|
|
||||||
|
if (canReportProgress)
|
||||||
{
|
{
|
||||||
isMoreToRead = false;
|
var percent = (int)(100.0 * totalRead / total);
|
||||||
}
|
//if (progressPercentage != percent && percent % 10 == 0)
|
||||||
else
|
|
||||||
{
|
|
||||||
var data = new byte[read];
|
|
||||||
buffer.ToList().CopyTo(0, data, 0, read);
|
|
||||||
|
|
||||||
// TODO: put here the code to write the file to disk
|
|
||||||
file.Write(data, 0, read);
|
|
||||||
|
|
||||||
totalRead += read;
|
|
||||||
|
|
||||||
if (canReportProgress)
|
|
||||||
{
|
{
|
||||||
var percent = Convert.ToInt32((totalRead * 1d) / (total * 1d) * 100);
|
progressPercentage = percent;
|
||||||
if (progressPercentage != percent && percent % 10 == 0)
|
progress!.Report(percent);
|
||||||
{
|
|
||||||
progressPercentage = percent;
|
|
||||||
progress.Report(percent);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} while (isMoreToRead);
|
}
|
||||||
if (canReportProgress)
|
if (canReportProgress)
|
||||||
{
|
{
|
||||||
progress.Report(101);
|
progress!.Report(101);
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task DownloadDataAsync4Speed(HttpClient client, string url, IProgress<string> progress, CancellationToken token)
|
public async Task DownloadDataAsync4Speed(HttpClient client, string url, IProgress<string> progress, CancellationToken token = default)
|
||||||
{
|
{
|
||||||
if (string.IsNullOrEmpty(url))
|
if (string.IsNullOrEmpty(url))
|
||||||
{
|
{
|
||||||
@@ -148,7 +97,7 @@ namespace v2rayN.Base
|
|||||||
|
|
||||||
if (!response.IsSuccessStatusCode)
|
if (!response.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
throw new Exception(string.Format("{0}", response.StatusCode));
|
throw new Exception(response.StatusCode.ToString());
|
||||||
}
|
}
|
||||||
|
|
||||||
//var total = response.Content.Headers.ContentLength.HasValue ? response.Content.Headers.ContentLength.Value : -1L;
|
//var total = response.Content.Headers.ContentLength.HasValue ? response.Content.Headers.ContentLength.Value : -1L;
|
||||||
|
|||||||
10
v2rayN/v2rayN/Base/MyDGTextColumn.cs
Normal file
10
v2rayN/v2rayN/Base/MyDGTextColumn.cs
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
using System.Windows.Controls;
|
||||||
|
|
||||||
|
namespace v2rayN.Base
|
||||||
|
{
|
||||||
|
internal class MyDGTextColumn : DataGridTextColumn
|
||||||
|
{
|
||||||
|
public string ExName { get; set; }
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -9,8 +9,9 @@ namespace v2rayN.Base
|
|||||||
private static readonly Lazy<SqliteHelper> _instance = new(() => new());
|
private static readonly Lazy<SqliteHelper> _instance = new(() => new());
|
||||||
public static SqliteHelper Instance => _instance.Value;
|
public static SqliteHelper Instance => _instance.Value;
|
||||||
private string _connstr;
|
private string _connstr;
|
||||||
public SQLiteConnection _db;
|
private SQLiteConnection _db;
|
||||||
public SQLiteAsyncConnection _dbAsync;
|
private SQLiteAsyncConnection _dbAsync;
|
||||||
|
private static readonly object objLock = new();
|
||||||
|
|
||||||
public SqliteHelper()
|
public SqliteHelper()
|
||||||
{
|
{
|
||||||
@@ -24,17 +25,27 @@ namespace v2rayN.Base
|
|||||||
return _db.CreateTable<T>();
|
return _db.CreateTable<T>();
|
||||||
}
|
}
|
||||||
|
|
||||||
public int Add(object model)
|
public int Insert(object model)
|
||||||
{
|
{
|
||||||
return _db.Insert(model);
|
return _db.Insert(model);
|
||||||
}
|
}
|
||||||
public async Task<int> AddAsync(object model)
|
public int InsertAll(IEnumerable models)
|
||||||
|
{
|
||||||
|
lock (objLock)
|
||||||
|
{
|
||||||
|
return _db.InsertAll(models);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public async Task<int> InsertAsync(object model)
|
||||||
{
|
{
|
||||||
return await _dbAsync.InsertAsync(model);
|
return await _dbAsync.InsertAsync(model);
|
||||||
}
|
}
|
||||||
public int Replace(object model)
|
public int Replace(object model)
|
||||||
{
|
{
|
||||||
return _db.InsertOrReplace(model);
|
lock (objLock)
|
||||||
|
{
|
||||||
|
return _db.InsertOrReplace(model);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
public async Task<int> Replacesync(object model)
|
public async Task<int> Replacesync(object model)
|
||||||
{
|
{
|
||||||
@@ -43,7 +54,10 @@ namespace v2rayN.Base
|
|||||||
|
|
||||||
public int Update(object model)
|
public int Update(object model)
|
||||||
{
|
{
|
||||||
return _db.Update(model);
|
lock (objLock)
|
||||||
|
{
|
||||||
|
return _db.Update(model);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
public async Task<int> UpdateAsync(object model)
|
public async Task<int> UpdateAsync(object model)
|
||||||
{
|
{
|
||||||
@@ -51,12 +65,18 @@ namespace v2rayN.Base
|
|||||||
}
|
}
|
||||||
public int UpdateAll(IEnumerable models)
|
public int UpdateAll(IEnumerable models)
|
||||||
{
|
{
|
||||||
return _db.UpdateAll(models);
|
lock (objLock)
|
||||||
|
{
|
||||||
|
return _db.UpdateAll(models);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public int Delete(object model)
|
public int Delete(object model)
|
||||||
{
|
{
|
||||||
return _db.Delete(model);
|
lock (objLock)
|
||||||
|
{
|
||||||
|
return _db.Delete(model);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
public async Task<int> DeleteAsync(object model)
|
public async Task<int> DeleteAsync(object model)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,15 +1,16 @@
|
|||||||
using System.IO;
|
using System.Diagnostics.CodeAnalysis;
|
||||||
|
using System.IO;
|
||||||
|
|
||||||
namespace v2rayN.Base
|
namespace v2rayN.Base
|
||||||
{
|
{
|
||||||
static class StringEx
|
static class StringEx
|
||||||
{
|
{
|
||||||
public static bool IsNullOrEmpty(this string value)
|
public static bool IsNullOrEmpty([NotNullWhen(false)] this string? value)
|
||||||
{
|
{
|
||||||
return string.IsNullOrEmpty(value);
|
return string.IsNullOrEmpty(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static bool IsNullOrWhiteSpace(this string value)
|
public static bool IsNullOrWhiteSpace([NotNullWhen(false)] this string? value)
|
||||||
{
|
{
|
||||||
return string.IsNullOrWhiteSpace(value);
|
return string.IsNullOrWhiteSpace(value);
|
||||||
}
|
}
|
||||||
@@ -42,7 +43,7 @@ namespace v2rayN.Base
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static string TrimEx(this string value)
|
public static string TrimEx(this string? value)
|
||||||
{
|
{
|
||||||
return value == null ? string.Empty : value.Trim();
|
return value == null ? string.Empty : value.Trim();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -33,6 +33,7 @@
|
|||||||
public const string CustomRoutingFileName = "v2rayN.Sample.custom_routing_";
|
public const string CustomRoutingFileName = "v2rayN.Sample.custom_routing_";
|
||||||
public const string v2raySampleInbound = "v2rayN.Sample.SampleInbound";
|
public const string v2raySampleInbound = "v2rayN.Sample.SampleInbound";
|
||||||
public const string TunSingboxFileName = "v2rayN.Sample.tun_singbox";
|
public const string TunSingboxFileName = "v2rayN.Sample.tun_singbox";
|
||||||
|
public const string TunSingboxDNSFileName = "v2rayN.Sample.tun_singbox_dns";
|
||||||
|
|
||||||
public const string DefaultSecurity = "auto";
|
public const string DefaultSecurity = "auto";
|
||||||
public const string DefaultNetwork = "tcp";
|
public const string DefaultNetwork = "tcp";
|
||||||
@@ -42,7 +43,7 @@
|
|||||||
public const string directTag = "direct";
|
public const string directTag = "direct";
|
||||||
public const string blockTag = "block";
|
public const string blockTag = "block";
|
||||||
public const string StreamSecurity = "tls";
|
public const string StreamSecurity = "tls";
|
||||||
public const string StreamSecurityX = "xtls";
|
public const string StreamSecurityReality = "reality";
|
||||||
public const string InboundSocks = "socks";
|
public const string InboundSocks = "socks";
|
||||||
public const string InboundHttp = "http";
|
public const string InboundHttp = "http";
|
||||||
public const string InboundSocks2 = "socks2";
|
public const string InboundSocks2 = "socks2";
|
||||||
@@ -91,7 +92,7 @@
|
|||||||
public static readonly List<string> ssSecuritys = new() { "aes-256-gcm", "aes-128-gcm", "chacha20-poly1305", "chacha20-ietf-poly1305", "none", "plain" };
|
public static readonly List<string> ssSecuritys = new() { "aes-256-gcm", "aes-128-gcm", "chacha20-poly1305", "chacha20-ietf-poly1305", "none", "plain" };
|
||||||
public static readonly List<string> ssSecuritysInSagerNet = new() { "none", "2022-blake3-aes-128-gcm", "2022-blake3-aes-256-gcm", "2022-blake3-chacha20-poly1305", "aes-128-gcm", "aes-192-gcm", "aes-256-gcm", "chacha20-ietf-poly1305", "xchacha20-ietf-poly1305", "rc4", "rc4-md5", "aes-128-ctr", "aes-192-ctr", "aes-256-ctr", "aes-128-cfb", "aes-192-cfb", "aes-256-cfb", "aes-128-cfb8", "aes-192-cfb8", "aes-256-cfb8", "aes-128-ofb", "aes-192-ofb", "aes-256-ofb", "bf-cfb", "cast5-cfb", "des-cfb", "idea-cfb", "rc2-cfb", "seed-cfb", "camellia-128-cfb", "camellia-192-cfb", "camellia-256-cfb", "camellia-128-cfb8", "camellia-192-cfb8", "camellia-256-cfb8", "salsa20", "chacha20", "chacha20-ietf", "xchacha20" };
|
public static readonly List<string> ssSecuritysInSagerNet = new() { "none", "2022-blake3-aes-128-gcm", "2022-blake3-aes-256-gcm", "2022-blake3-chacha20-poly1305", "aes-128-gcm", "aes-192-gcm", "aes-256-gcm", "chacha20-ietf-poly1305", "xchacha20-ietf-poly1305", "rc4", "rc4-md5", "aes-128-ctr", "aes-192-ctr", "aes-256-ctr", "aes-128-cfb", "aes-192-cfb", "aes-256-cfb", "aes-128-cfb8", "aes-192-cfb8", "aes-256-cfb8", "aes-128-ofb", "aes-192-ofb", "aes-256-ofb", "bf-cfb", "cast5-cfb", "des-cfb", "idea-cfb", "rc2-cfb", "seed-cfb", "camellia-128-cfb", "camellia-192-cfb", "camellia-256-cfb", "camellia-128-cfb8", "camellia-192-cfb8", "camellia-256-cfb8", "salsa20", "chacha20", "chacha20-ietf", "xchacha20" };
|
||||||
public static readonly List<string> ssSecuritysInXray = new() { "aes-256-gcm", "aes-128-gcm", "chacha20-poly1305", "chacha20-ietf-poly1305", "xchacha20-poly1305", "xchacha20-ietf-poly1305", "none", "plain", "2022-blake3-aes-128-gcm", "2022-blake3-aes-256-gcm", "2022-blake3-chacha20-poly1305" };
|
public static readonly List<string> ssSecuritysInXray = new() { "aes-256-gcm", "aes-128-gcm", "chacha20-poly1305", "chacha20-ietf-poly1305", "xchacha20-poly1305", "xchacha20-ietf-poly1305", "none", "plain", "2022-blake3-aes-128-gcm", "2022-blake3-aes-256-gcm", "2022-blake3-chacha20-poly1305" };
|
||||||
public static readonly List<string> xtlsFlows = new() { "", "xtls-rprx-origin", "xtls-rprx-origin-udp443", "xtls-rprx-direct", "xtls-rprx-direct-udp443", "xtls-rprx-vision", "xtls-rprx-vision-udp443" };
|
public static readonly List<string> flows = new() { "", "xtls-rprx-vision", "xtls-rprx-vision-udp443" };
|
||||||
public static readonly List<string> networks = new() { "tcp", "kcp", "ws", "h2", "quic", "grpc" };
|
public static readonly List<string> networks = new() { "tcp", "kcp", "ws", "h2", "quic", "grpc" };
|
||||||
public static readonly List<string> kcpHeaderTypes = new() { "srtp", "utp", "wechat-video", "dtls", "wireguard" };
|
public static readonly List<string> kcpHeaderTypes = new() { "srtp", "utp", "wechat-video", "dtls", "wireguard" };
|
||||||
public static readonly List<string> coreTypes = new() { "v2fly", "SagerNet", "Xray", "v2fly_v5" };
|
public static readonly List<string> coreTypes = new() { "v2fly", "SagerNet", "Xray", "v2fly_v5" };
|
||||||
@@ -123,7 +124,6 @@
|
|||||||
|
|
||||||
#region global variable
|
#region global variable
|
||||||
|
|
||||||
public static bool reloadCore { get; set; }
|
|
||||||
public static int statePort { get; set; }
|
public static int statePort { get; set; }
|
||||||
public static Job processJob { get; set; }
|
public static Job processJob { get; set; }
|
||||||
public static bool ShowInTaskbar { get; set; }
|
public static bool ShowInTaskbar { get; set; }
|
||||||
|
|||||||
@@ -151,9 +151,9 @@ namespace v2rayN.Handler
|
|||||||
enableAutoAdjustMainLvColWidth = true
|
enableAutoAdjustMainLvColWidth = true
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
if (config.uiItem.mainLvColWidth == null)
|
if (config.uiItem.mainColumnItem == null)
|
||||||
{
|
{
|
||||||
config.uiItem.mainLvColWidth = new Dictionary<string, int>();
|
config.uiItem.mainColumnItem = new();
|
||||||
}
|
}
|
||||||
if (Utils.IsNullOrEmpty(config.uiItem.currentLanguage))
|
if (Utils.IsNullOrEmpty(config.uiItem.currentLanguage))
|
||||||
{
|
{
|
||||||
@@ -195,16 +195,6 @@ namespace v2rayN.Handler
|
|||||||
config.guiItem.statisticsFreshRate = 1;
|
config.guiItem.statisticsFreshRate = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (config == null)
|
|
||||||
{
|
|
||||||
Global.reloadCore = false;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Global.reloadCore = true;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
LazyConfig.Instance.SetConfig(config);
|
LazyConfig.Instance.SetConfig(config);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -215,7 +205,6 @@ namespace v2rayN.Handler
|
|||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static int SaveConfig(ref Config config, bool reload = true)
|
public static int SaveConfig(ref Config config, bool reload = true)
|
||||||
{
|
{
|
||||||
Global.reloadCore = reload;
|
|
||||||
|
|
||||||
ToJsonFile(config);
|
ToJsonFile(config);
|
||||||
|
|
||||||
@@ -363,7 +352,7 @@ namespace v2rayN.Handler
|
|||||||
/// <param name="config"></param>
|
/// <param name="config"></param>
|
||||||
/// <param name="profileItem"></param>
|
/// <param name="profileItem"></param>
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static int AddServer(ref Config config, ProfileItem profileItem)
|
public static int AddServer(ref Config config, ProfileItem profileItem, bool toFile = true)
|
||||||
{
|
{
|
||||||
profileItem.configType = EConfigType.VMess;
|
profileItem.configType = EConfigType.VMess;
|
||||||
|
|
||||||
@@ -381,7 +370,7 @@ namespace v2rayN.Handler
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
AddServerCommon(ref config, profileItem);
|
AddServerCommon(ref config, profileItem, toFile);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -394,11 +383,15 @@ namespace v2rayN.Handler
|
|||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static int RemoveServer(Config config, List<ProfileItem> indexs)
|
public static int RemoveServer(Config config, List<ProfileItem> indexs)
|
||||||
{
|
{
|
||||||
|
var subid = "TempRemoveSubId";
|
||||||
foreach (var item in indexs)
|
foreach (var item in indexs)
|
||||||
{
|
{
|
||||||
RemoveProfileItem(config, item.indexId);
|
item.subid = subid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SqliteHelper.Instance.UpdateAll(indexs);
|
||||||
|
RemoveServerViaSubid(ref config, subid, false);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -410,8 +403,14 @@ namespace v2rayN.Handler
|
|||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static int CopyServer(ref Config config, List<ProfileItem> indexs)
|
public static int CopyServer(ref Config config, List<ProfileItem> indexs)
|
||||||
{
|
{
|
||||||
foreach (var item in indexs)
|
foreach (var it in indexs)
|
||||||
{
|
{
|
||||||
|
var item = LazyConfig.Instance.GetProfileItem(it.indexId);
|
||||||
|
if (item is null)
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
ProfileItem profileItem = Utils.DeepCopy(item);
|
ProfileItem profileItem = Utils.DeepCopy(item);
|
||||||
profileItem.indexId = string.Empty;
|
profileItem.indexId = string.Empty;
|
||||||
profileItem.remarks = $"{item.remarks}-clone";
|
profileItem.remarks = $"{item.remarks}-clone";
|
||||||
@@ -425,7 +424,7 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
AddServerCommon(ref config, profileItem);
|
AddServerCommon(ref config, profileItem, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -446,7 +445,6 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
|
|
||||||
config.indexId = indexId;
|
config.indexId = indexId;
|
||||||
Global.reloadCore = true;
|
|
||||||
|
|
||||||
ToJsonFile(config);
|
ToJsonFile(config);
|
||||||
|
|
||||||
@@ -459,8 +457,7 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
var allItems = LazyConfig.Instance.ProfileItemIndexs("");
|
if (SqliteHelper.Instance.Table<ProfileItem>().Where(t => t.indexId == config.indexId).Any())
|
||||||
if (allItems.Where(t => t == config.indexId).Any())
|
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -468,11 +465,7 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
return SetDefaultServerIndex(ref config, lstProfile[0].indexId);
|
return SetDefaultServerIndex(ref config, lstProfile[0].indexId);
|
||||||
}
|
}
|
||||||
if (allItems.Count > 0)
|
return SetDefaultServerIndex(ref config, SqliteHelper.Instance.Table<ProfileItem>().Select(t => t.indexId).FirstOrDefault());
|
||||||
{
|
|
||||||
return SetDefaultServerIndex(ref config, allItems.FirstOrDefault());
|
|
||||||
}
|
|
||||||
return -1;
|
|
||||||
}
|
}
|
||||||
public static ProfileItem? GetDefaultServer(ref Config config)
|
public static ProfileItem? GetDefaultServer(ref Config config)
|
||||||
{
|
{
|
||||||
@@ -600,7 +593,7 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
AddServerCommon(ref config, profileItem);
|
AddServerCommon(ref config, profileItem, true);
|
||||||
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@@ -632,7 +625,7 @@ namespace v2rayN.Handler
|
|||||||
/// <param name="config"></param>
|
/// <param name="config"></param>
|
||||||
/// <param name="profileItem"></param>
|
/// <param name="profileItem"></param>
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static int AddShadowsocksServer(ref Config config, ProfileItem profileItem)
|
public static int AddShadowsocksServer(ref Config config, ProfileItem profileItem, bool toFile = true)
|
||||||
{
|
{
|
||||||
profileItem.configType = EConfigType.Shadowsocks;
|
profileItem.configType = EConfigType.Shadowsocks;
|
||||||
|
|
||||||
@@ -645,7 +638,7 @@ namespace v2rayN.Handler
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
AddServerCommon(ref config, profileItem);
|
AddServerCommon(ref config, profileItem, toFile);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -656,13 +649,13 @@ namespace v2rayN.Handler
|
|||||||
/// <param name="config"></param>
|
/// <param name="config"></param>
|
||||||
/// <param name="profileItem"></param>
|
/// <param name="profileItem"></param>
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static int AddSocksServer(ref Config config, ProfileItem profileItem)
|
public static int AddSocksServer(ref Config config, ProfileItem profileItem, bool toFile = true)
|
||||||
{
|
{
|
||||||
profileItem.configType = EConfigType.Socks;
|
profileItem.configType = EConfigType.Socks;
|
||||||
|
|
||||||
profileItem.address = profileItem.address.TrimEx();
|
profileItem.address = profileItem.address.TrimEx();
|
||||||
|
|
||||||
AddServerCommon(ref config, profileItem);
|
AddServerCommon(ref config, profileItem, toFile);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -673,7 +666,7 @@ namespace v2rayN.Handler
|
|||||||
/// <param name="config"></param>
|
/// <param name="config"></param>
|
||||||
/// <param name="profileItem"></param>
|
/// <param name="profileItem"></param>
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static int AddTrojanServer(ref Config config, ProfileItem profileItem)
|
public static int AddTrojanServer(ref Config config, ProfileItem profileItem, bool toFile = true)
|
||||||
{
|
{
|
||||||
profileItem.configType = EConfigType.Trojan;
|
profileItem.configType = EConfigType.Trojan;
|
||||||
|
|
||||||
@@ -683,18 +676,14 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
profileItem.streamSecurity = Global.StreamSecurity;
|
profileItem.streamSecurity = Global.StreamSecurity;
|
||||||
}
|
}
|
||||||
if (Utils.IsNullOrEmpty(profileItem.allowInsecure))
|
|
||||||
{
|
|
||||||
profileItem.allowInsecure = config.coreBasicItem.defAllowInsecure.ToString().ToLower();
|
|
||||||
}
|
|
||||||
|
|
||||||
AddServerCommon(ref config, profileItem);
|
AddServerCommon(ref config, profileItem, toFile);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public static int SortServers(ref Config config, string subId, EServerColName name, bool asc)
|
public static int SortServers(ref Config config, string subId, string colName, bool asc)
|
||||||
{
|
{
|
||||||
var lstModel = LazyConfig.Instance.ProfileItems(subId, "");
|
var lstModel = LazyConfig.Instance.ProfileItems(subId, "");
|
||||||
if (lstModel.Count <= 0)
|
if (lstModel.Count <= 0)
|
||||||
@@ -720,6 +709,8 @@ namespace v2rayN.Handler
|
|||||||
sort = t33 == null ? 0 : t33.sort
|
sort = t33 == null ? 0 : t33.sort
|
||||||
}).ToList();
|
}).ToList();
|
||||||
|
|
||||||
|
|
||||||
|
Enum.TryParse(colName, true, out EServerColName name);
|
||||||
var propertyName = string.Empty;
|
var propertyName = string.Empty;
|
||||||
switch (name)
|
switch (name)
|
||||||
{
|
{
|
||||||
@@ -730,10 +721,14 @@ namespace v2rayN.Handler
|
|||||||
case EServerColName.security:
|
case EServerColName.security:
|
||||||
case EServerColName.network:
|
case EServerColName.network:
|
||||||
case EServerColName.streamSecurity:
|
case EServerColName.streamSecurity:
|
||||||
case EServerColName.delay:
|
|
||||||
case EServerColName.speed:
|
|
||||||
propertyName = name.ToString();
|
propertyName = name.ToString();
|
||||||
break;
|
break;
|
||||||
|
case EServerColName.delayVal:
|
||||||
|
propertyName = "delay";
|
||||||
|
break;
|
||||||
|
case EServerColName.speedVal:
|
||||||
|
propertyName = "speed";
|
||||||
|
break;
|
||||||
case EServerColName.subRemarks:
|
case EServerColName.subRemarks:
|
||||||
propertyName = "subid";
|
propertyName = "subid";
|
||||||
break;
|
break;
|
||||||
@@ -755,7 +750,7 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
ProfileExHandler.Instance.SetSort(lstProfile[i].indexId, (i + 1) * 10);
|
ProfileExHandler.Instance.SetSort(lstProfile[i].indexId, (i + 1) * 10);
|
||||||
}
|
}
|
||||||
if (name == EServerColName.delay)
|
if (name == EServerColName.delayVal)
|
||||||
{
|
{
|
||||||
var maxSort = lstProfile.Max(t => t.sort) + 10;
|
var maxSort = lstProfile.Max(t => t.sort) + 10;
|
||||||
foreach (var item in lstProfile)
|
foreach (var item in lstProfile)
|
||||||
@@ -766,7 +761,7 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (name == EServerColName.speed)
|
if (name == EServerColName.speedVal)
|
||||||
{
|
{
|
||||||
var maxSort = lstProfile.Max(t => t.sort) + 10;
|
var maxSort = lstProfile.Max(t => t.sort) + 10;
|
||||||
foreach (var item in lstProfile)
|
foreach (var item in lstProfile)
|
||||||
@@ -787,7 +782,7 @@ namespace v2rayN.Handler
|
|||||||
/// <param name="config"></param>
|
/// <param name="config"></param>
|
||||||
/// <param name="profileItem"></param>
|
/// <param name="profileItem"></param>
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static int AddVlessServer(ref Config config, ProfileItem profileItem)
|
public static int AddVlessServer(ref Config config, ProfileItem profileItem, bool toFile = true)
|
||||||
{
|
{
|
||||||
profileItem.configType = EConfigType.VLESS;
|
profileItem.configType = EConfigType.VLESS;
|
||||||
|
|
||||||
@@ -800,43 +795,51 @@ namespace v2rayN.Handler
|
|||||||
profileItem.path = profileItem.path.TrimEx();
|
profileItem.path = profileItem.path.TrimEx();
|
||||||
profileItem.streamSecurity = profileItem.streamSecurity.TrimEx();
|
profileItem.streamSecurity = profileItem.streamSecurity.TrimEx();
|
||||||
|
|
||||||
AddServerCommon(ref config, profileItem);
|
AddServerCommon(ref config, profileItem, toFile);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static int DedupServerList(ref Config config, ref List<ProfileItem> lstProfile)
|
public static Tuple<int, int> DedupServerList(Config config, string subId)
|
||||||
{
|
{
|
||||||
List<ProfileItem> source = lstProfile;
|
var lstProfile = LazyConfig.Instance.ProfileItems(subId);
|
||||||
bool keepOlder = config.guiItem.keepOlderDedupl;
|
|
||||||
|
|
||||||
List<ProfileItem> list = new();
|
List<ProfileItem> lstKeep = new();
|
||||||
if (!keepOlder) source.Reverse(); // Remove the early items first
|
List<ProfileItem> lstRemove = new();
|
||||||
|
if (config.guiItem.keepOlderDedupl) lstProfile.Reverse();
|
||||||
|
|
||||||
foreach (ProfileItem item in source)
|
foreach (ProfileItem item in lstProfile)
|
||||||
{
|
{
|
||||||
if (!list.Exists(i => CompareProfileItem(i, item, false)))
|
if (!lstKeep.Exists(i => CompareProfileItem(i, item, false)))
|
||||||
{
|
{
|
||||||
list.Add(item);
|
lstKeep.Add(item);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
RemoveProfileItem(config, item.indexId);
|
lstRemove.Add(item);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
//if (!keepOlder) list.Reverse();
|
RemoveServer(config, lstRemove);
|
||||||
//config.vmess = list;
|
|
||||||
|
|
||||||
return list.Count;
|
return new Tuple<int, int>(lstProfile.Count, lstKeep.Count);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static int AddServerCommon(ref Config config, ProfileItem profileItem)
|
public static int AddServerCommon(ref Config config, ProfileItem profileItem, bool toFile = true)
|
||||||
{
|
{
|
||||||
profileItem.configVersion = 2;
|
profileItem.configVersion = 2;
|
||||||
if (Utils.IsNullOrEmpty(profileItem.allowInsecure))
|
|
||||||
|
if (!Utils.IsNullOrEmpty(profileItem.streamSecurity))
|
||||||
{
|
{
|
||||||
profileItem.allowInsecure = config.coreBasicItem.defAllowInsecure.ToString().ToLower();
|
if (Utils.IsNullOrEmpty(profileItem.allowInsecure))
|
||||||
|
{
|
||||||
|
profileItem.allowInsecure = config.coreBasicItem.defAllowInsecure.ToString().ToLower();
|
||||||
|
}
|
||||||
|
if (Utils.IsNullOrEmpty(profileItem.fingerprint))
|
||||||
|
{
|
||||||
|
profileItem.fingerprint = config.coreBasicItem.defFingerprint;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!Utils.IsNullOrEmpty(profileItem.network) && !Global.networks.Contains(profileItem.network))
|
if (!Utils.IsNullOrEmpty(profileItem.network) && !Global.networks.Contains(profileItem.network))
|
||||||
{
|
{
|
||||||
profileItem.network = Global.DefaultNetwork;
|
profileItem.network = Global.DefaultNetwork;
|
||||||
@@ -848,19 +851,12 @@ namespace v2rayN.Handler
|
|||||||
var maxSort = ProfileExHandler.Instance.GetMaxSort();
|
var maxSort = ProfileExHandler.Instance.GetMaxSort();
|
||||||
ProfileExHandler.Instance.SetSort(profileItem.indexId, maxSort + 1);
|
ProfileExHandler.Instance.SetSort(profileItem.indexId, maxSort + 1);
|
||||||
}
|
}
|
||||||
else if (profileItem.indexId == config.indexId)
|
|
||||||
{
|
|
||||||
Global.reloadCore = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (SqliteHelper.Instance.Replace(profileItem) > 0)
|
if (toFile)
|
||||||
{
|
{
|
||||||
return 0;
|
SqliteHelper.Instance.Replace(profileItem);
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
return -1;
|
|
||||||
}
|
}
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static bool CompareProfileItem(ProfileItem o, ProfileItem n, bool remarks)
|
private static bool CompareProfileItem(ProfileItem o, ProfileItem n, bool remarks)
|
||||||
@@ -937,11 +933,7 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
|
|
||||||
int countServers = 0;
|
int countServers = 0;
|
||||||
//var maxSort = 0;
|
List<ProfileItem> lstAdd = new();
|
||||||
//if (SqliteHelper.Instance.Table<ProfileItem>().Count() > 0)
|
|
||||||
//{
|
|
||||||
// maxSort = SqliteHelper.Instance.Table<ProfileItem>().Max(t => t.sort);
|
|
||||||
//}
|
|
||||||
string[] arrData = clipboardData.Split(Environment.NewLine.ToCharArray());
|
string[] arrData = clipboardData.Split(Environment.NewLine.ToCharArray());
|
||||||
foreach (string str in arrData)
|
foreach (string str in arrData)
|
||||||
{
|
{
|
||||||
@@ -963,10 +955,10 @@ namespace v2rayN.Handler
|
|||||||
//exist sub items
|
//exist sub items
|
||||||
if (isSub && !Utils.IsNullOrEmpty(subid))
|
if (isSub && !Utils.IsNullOrEmpty(subid))
|
||||||
{
|
{
|
||||||
var existItem = lstOriSub?.FirstOrDefault(t => CompareProfileItem(t, profileItem, true));
|
var existItem = lstOriSub?.FirstOrDefault(t => t.isSub == isSub && CompareProfileItem(t, profileItem, true));
|
||||||
if (existItem != null)
|
if (existItem != null)
|
||||||
{
|
{
|
||||||
profileItem = existItem;
|
profileItem.indexId = existItem.indexId;
|
||||||
}
|
}
|
||||||
//filter
|
//filter
|
||||||
if (!Utils.IsNullOrEmpty(subFilter))
|
if (!Utils.IsNullOrEmpty(subFilter))
|
||||||
@@ -979,43 +971,39 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
profileItem.subid = subid;
|
profileItem.subid = subid;
|
||||||
profileItem.isSub = isSub;
|
profileItem.isSub = isSub;
|
||||||
//profileItem.sort = maxSort + countServers + 1;
|
var addStatus = -1;
|
||||||
|
|
||||||
if (profileItem.configType == EConfigType.VMess)
|
if (profileItem.configType == EConfigType.VMess)
|
||||||
{
|
{
|
||||||
if (AddServer(ref config, profileItem) == 0)
|
addStatus = AddServer(ref config, profileItem, false);
|
||||||
{
|
|
||||||
countServers++;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else if (profileItem.configType == EConfigType.Shadowsocks)
|
else if (profileItem.configType == EConfigType.Shadowsocks)
|
||||||
{
|
{
|
||||||
if (AddShadowsocksServer(ref config, profileItem) == 0)
|
addStatus = AddShadowsocksServer(ref config, profileItem, false);
|
||||||
{
|
|
||||||
countServers++;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else if (profileItem.configType == EConfigType.Socks)
|
else if (profileItem.configType == EConfigType.Socks)
|
||||||
{
|
{
|
||||||
if (AddSocksServer(ref config, profileItem) == 0)
|
addStatus = AddSocksServer(ref config, profileItem, false);
|
||||||
{
|
|
||||||
countServers++;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else if (profileItem.configType == EConfigType.Trojan)
|
else if (profileItem.configType == EConfigType.Trojan)
|
||||||
{
|
{
|
||||||
if (AddTrojanServer(ref config, profileItem) == 0)
|
addStatus = AddTrojanServer(ref config, profileItem, false);
|
||||||
{
|
|
||||||
countServers++;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else if (profileItem.configType == EConfigType.VLESS)
|
else if (profileItem.configType == EConfigType.VLESS)
|
||||||
{
|
{
|
||||||
if (AddVlessServer(ref config, profileItem) == 0)
|
addStatus = AddVlessServer(ref config, profileItem, false);
|
||||||
{
|
|
||||||
countServers++;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (addStatus == 0)
|
||||||
|
{
|
||||||
|
countServers++;
|
||||||
|
lstAdd.Add(profileItem);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (lstAdd.Count > 0)
|
||||||
|
{
|
||||||
|
SqliteHelper.Instance.InsertAll(lstAdd);
|
||||||
}
|
}
|
||||||
|
|
||||||
ToJsonFile(config);
|
ToJsonFile(config);
|
||||||
@@ -1042,11 +1030,8 @@ namespace v2rayN.Handler
|
|||||||
ProfileItem profileItem = new();
|
ProfileItem profileItem = new();
|
||||||
//Is v2ray configuration
|
//Is v2ray configuration
|
||||||
V2rayConfig? v2rayConfig = Utils.FromJson<V2rayConfig>(clipboardData);
|
V2rayConfig? v2rayConfig = Utils.FromJson<V2rayConfig>(clipboardData);
|
||||||
if (v2rayConfig != null
|
if (v2rayConfig?.inbounds?.Count > 0
|
||||||
&& v2rayConfig.inbounds != null
|
&& v2rayConfig.outbounds?.Count > 0)
|
||||||
&& v2rayConfig.inbounds.Count > 0
|
|
||||||
&& v2rayConfig.outbounds != null
|
|
||||||
&& v2rayConfig.outbounds.Count > 0)
|
|
||||||
{
|
{
|
||||||
var fileName = Utils.GetTempPath($"{Utils.GetGUID(false)}.json");
|
var fileName = Utils.GetTempPath($"{Utils.GetGUID(false)}.json");
|
||||||
File.WriteAllText(fileName, clipboardData);
|
File.WriteAllText(fileName, clipboardData);
|
||||||
@@ -1100,7 +1085,7 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
RemoveServerViaSubid(ref config, subid, isSub);
|
RemoveServerViaSubid(ref config, subid, isSub);
|
||||||
}
|
}
|
||||||
if (isSub && lstOriSub != null && lstOriSub.Count == 1)
|
if (isSub && lstOriSub?.Count == 1)
|
||||||
{
|
{
|
||||||
profileItem.indexId = lstOriSub[0].indexId;
|
profileItem.indexId = lstOriSub[0].indexId;
|
||||||
}
|
}
|
||||||
@@ -1137,16 +1122,16 @@ namespace v2rayN.Handler
|
|||||||
|
|
||||||
//SsSIP008
|
//SsSIP008
|
||||||
var lstSsServer = Utils.FromJson<List<SsServer>>(clipboardData);
|
var lstSsServer = Utils.FromJson<List<SsServer>>(clipboardData);
|
||||||
if (lstSsServer == null || lstSsServer.Count <= 0)
|
if (lstSsServer?.Count <= 0)
|
||||||
{
|
{
|
||||||
var ssSIP008 = Utils.FromJson<SsSIP008>(clipboardData);
|
var ssSIP008 = Utils.FromJson<SsSIP008>(clipboardData);
|
||||||
if (ssSIP008?.servers != null && ssSIP008.servers.Count > 0)
|
if (ssSIP008?.servers?.Count > 0)
|
||||||
{
|
{
|
||||||
lstSsServer = ssSIP008.servers;
|
lstSsServer = ssSIP008.servers;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (lstSsServer != null && lstSsServer.Count > 0)
|
if (lstSsServer?.Count > 0)
|
||||||
{
|
{
|
||||||
int counter = 0;
|
int counter = 0;
|
||||||
foreach (var it in lstSsServer)
|
foreach (var it in lstSsServer)
|
||||||
@@ -1270,6 +1255,7 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
var customProfile = SqliteHelper.Instance.Table<ProfileItem>().Where(t => t.subid == subid && t.configType == EConfigType.Custom).ToList();
|
||||||
if (isSub)
|
if (isSub)
|
||||||
{
|
{
|
||||||
SqliteHelper.Instance.Execute($"delete from ProfileItem where isSub = 1 and subid = '{subid}'");
|
SqliteHelper.Instance.Execute($"delete from ProfileItem where isSub = 1 and subid = '{subid}'");
|
||||||
@@ -1278,6 +1264,10 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
SqliteHelper.Instance.Execute($"delete from ProfileItem where subid = '{subid}'");
|
SqliteHelper.Instance.Execute($"delete from ProfileItem where subid = '{subid}'");
|
||||||
}
|
}
|
||||||
|
foreach (var item in customProfile)
|
||||||
|
{
|
||||||
|
File.Delete(Utils.GetConfigPath(item.address));
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -1313,44 +1303,6 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region UI
|
|
||||||
|
|
||||||
public static int AddformMainLvColWidth(ref Config config, string name, int width)
|
|
||||||
{
|
|
||||||
if (config.uiItem.mainLvColWidth == null)
|
|
||||||
{
|
|
||||||
config.uiItem.mainLvColWidth = new Dictionary<string, int>();
|
|
||||||
}
|
|
||||||
if (config.uiItem.mainLvColWidth.ContainsKey(name))
|
|
||||||
{
|
|
||||||
config.uiItem.mainLvColWidth[name] = width;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
config.uiItem.mainLvColWidth.Add(name, width);
|
|
||||||
}
|
|
||||||
|
|
||||||
ToJsonFile(config);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
public static int GetformMainLvColWidth(ref Config config, string name, int width)
|
|
||||||
{
|
|
||||||
if (config.uiItem.mainLvColWidth == null)
|
|
||||||
{
|
|
||||||
config.uiItem.mainLvColWidth = new Dictionary<string, int>();
|
|
||||||
}
|
|
||||||
if (config.uiItem.mainLvColWidth.ContainsKey(name))
|
|
||||||
{
|
|
||||||
return config.uiItem.mainLvColWidth[name];
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
return width;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region Routing
|
#region Routing
|
||||||
|
|
||||||
public static int SaveRoutingItem(ref Config config, RoutingItem item)
|
public static int SaveRoutingItem(ref Config config, RoutingItem item)
|
||||||
@@ -1496,8 +1448,6 @@ namespace v2rayN.Handler
|
|||||||
config.routingBasicItem.routingIndexId = routingItem.id;
|
config.routingBasicItem.routingIndexId = routingItem.id;
|
||||||
}
|
}
|
||||||
|
|
||||||
Global.reloadCore = true;
|
|
||||||
|
|
||||||
ToJsonFile(config);
|
ToJsonFile(config);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|||||||
@@ -184,8 +184,7 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (v2rayConfig.routing != null
|
if (v2rayConfig.routing?.rules != null)
|
||||||
&& v2rayConfig.routing.rules != null)
|
|
||||||
{
|
{
|
||||||
v2rayConfig.routing.domainStrategy = config.routingBasicItem.domainStrategy;
|
v2rayConfig.routing.domainStrategy = config.routingBasicItem.domainStrategy;
|
||||||
v2rayConfig.routing.domainMatcher = Utils.IsNullOrEmpty(config.routingBasicItem.domainMatcher) ? null : config.routingBasicItem.domainMatcher;
|
v2rayConfig.routing.domainMatcher = Utils.IsNullOrEmpty(config.routingBasicItem.domainMatcher) ? null : config.routingBasicItem.domainMatcher;
|
||||||
@@ -241,25 +240,25 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
rules.port = null;
|
rules.port = null;
|
||||||
}
|
}
|
||||||
if (rules.domain != null && rules.domain.Count == 0)
|
if (rules.domain?.Count == 0)
|
||||||
{
|
{
|
||||||
rules.domain = null;
|
rules.domain = null;
|
||||||
}
|
}
|
||||||
if (rules.ip != null && rules.ip.Count == 0)
|
if (rules.ip?.Count == 0)
|
||||||
{
|
{
|
||||||
rules.ip = null;
|
rules.ip = null;
|
||||||
}
|
}
|
||||||
if (rules.protocol != null && rules.protocol.Count == 0)
|
if (rules.protocol?.Count == 0)
|
||||||
{
|
{
|
||||||
rules.protocol = null;
|
rules.protocol = null;
|
||||||
}
|
}
|
||||||
if (rules.inboundTag != null && rules.inboundTag.Count == 0)
|
if (rules.inboundTag?.Count == 0)
|
||||||
{
|
{
|
||||||
rules.inboundTag = null;
|
rules.inboundTag = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
var hasDomainIp = false;
|
var hasDomainIp = false;
|
||||||
if (rules.domain != null && rules.domain.Count > 0)
|
if (rules.domain?.Count > 0)
|
||||||
{
|
{
|
||||||
var it = Utils.DeepCopy(rules);
|
var it = Utils.DeepCopy(rules);
|
||||||
it.ip = null;
|
it.ip = null;
|
||||||
@@ -275,7 +274,7 @@ namespace v2rayN.Handler
|
|||||||
v2rayConfig.routing.rules.Add(it);
|
v2rayConfig.routing.rules.Add(it);
|
||||||
hasDomainIp = true;
|
hasDomainIp = true;
|
||||||
}
|
}
|
||||||
if (rules.ip != null && rules.ip.Count > 0)
|
if (rules.ip?.Count > 0)
|
||||||
{
|
{
|
||||||
var it = Utils.DeepCopy(rules);
|
var it = Utils.DeepCopy(rules);
|
||||||
it.domain = null;
|
it.domain = null;
|
||||||
@@ -286,8 +285,8 @@ namespace v2rayN.Handler
|
|||||||
if (!hasDomainIp)
|
if (!hasDomainIp)
|
||||||
{
|
{
|
||||||
if (!Utils.IsNullOrEmpty(rules.port)
|
if (!Utils.IsNullOrEmpty(rules.port)
|
||||||
|| (rules.protocol != null && rules.protocol.Count > 0)
|
|| (rules.protocol?.Count > 0)
|
||||||
|| (rules.inboundTag != null && rules.inboundTag.Count > 0)
|
|| (rules.inboundTag?.Count > 0)
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
var it = Utils.DeepCopy(rules);
|
var it = Utils.DeepCopy(rules);
|
||||||
@@ -457,17 +456,9 @@ namespace v2rayN.Handler
|
|||||||
|
|
||||||
boundStreamSettings(node, "out", outbound.streamSettings);
|
boundStreamSettings(node, "out", outbound.streamSettings);
|
||||||
|
|
||||||
//if xtls
|
if (node.streamSecurity == Global.StreamSecurityReality)
|
||||||
if (node.streamSecurity == Global.StreamSecurityX)
|
|
||||||
{
|
{
|
||||||
if (Utils.IsNullOrEmpty(node.flow))
|
usersItem.flow = node.flow;
|
||||||
{
|
|
||||||
usersItem.flow = Global.xtlsFlows[1];
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
usersItem.flow = node.flow.Replace("splice", "direct");
|
|
||||||
}
|
|
||||||
|
|
||||||
outbound.mux.enabled = false;
|
outbound.mux.enabled = false;
|
||||||
outbound.mux.concurrency = -1;
|
outbound.mux.concurrency = -1;
|
||||||
@@ -506,22 +497,6 @@ namespace v2rayN.Handler
|
|||||||
serversItem.ota = false;
|
serversItem.ota = false;
|
||||||
serversItem.level = 1;
|
serversItem.level = 1;
|
||||||
|
|
||||||
//if xtls
|
|
||||||
if (node.streamSecurity == Global.StreamSecurityX)
|
|
||||||
{
|
|
||||||
if (Utils.IsNullOrEmpty(node.flow))
|
|
||||||
{
|
|
||||||
serversItem.flow = Global.xtlsFlows[1];
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
serversItem.flow = node.flow.Replace("splice", "direct");
|
|
||||||
}
|
|
||||||
|
|
||||||
outbound.mux.enabled = false;
|
|
||||||
outbound.mux.concurrency = -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
outbound.mux.enabled = false;
|
outbound.mux.enabled = false;
|
||||||
outbound.mux.concurrency = -1;
|
outbound.mux.concurrency = -1;
|
||||||
|
|
||||||
@@ -582,26 +557,21 @@ namespace v2rayN.Handler
|
|||||||
streamSettings.tlsSettings = tlsSettings;
|
streamSettings.tlsSettings = tlsSettings;
|
||||||
}
|
}
|
||||||
|
|
||||||
//if xtls
|
//if Reality
|
||||||
if (node.streamSecurity == Global.StreamSecurityX)
|
if (node.streamSecurity == Global.StreamSecurityReality)
|
||||||
{
|
{
|
||||||
streamSettings.security = node.streamSecurity;
|
streamSettings.security = node.streamSecurity;
|
||||||
|
|
||||||
TlsSettings xtlsSettings = new()
|
TlsSettings realitySettings = new()
|
||||||
{
|
{
|
||||||
allowInsecure = Utils.ToBool(node.allowInsecure.IsNullOrEmpty() ? config.coreBasicItem.defAllowInsecure.ToString().ToLower() : node.allowInsecure),
|
fingerprint = node.fingerprint.IsNullOrEmpty() ? config.coreBasicItem.defFingerprint : node.fingerprint,
|
||||||
alpn = node.GetAlpn(),
|
serverName = sni,
|
||||||
fingerprint = node.fingerprint.IsNullOrEmpty() ? config.coreBasicItem.defFingerprint : node.fingerprint
|
publicKey = node.publicKey,
|
||||||
|
shortId = node.shortId,
|
||||||
|
spiderX = node.spiderX,
|
||||||
};
|
};
|
||||||
if (!string.IsNullOrWhiteSpace(sni))
|
|
||||||
{
|
streamSettings.realitySettings = realitySettings;
|
||||||
xtlsSettings.serverName = sni;
|
|
||||||
}
|
|
||||||
else if (!string.IsNullOrWhiteSpace(host))
|
|
||||||
{
|
|
||||||
xtlsSettings.serverName = Utils.String2List(host)[0];
|
|
||||||
}
|
|
||||||
streamSettings.xtlsSettings = xtlsSettings;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//streamSettings
|
//streamSettings
|
||||||
@@ -613,12 +583,12 @@ namespace v2rayN.Handler
|
|||||||
mtu = config.kcpItem.mtu,
|
mtu = config.kcpItem.mtu,
|
||||||
tti = config.kcpItem.tti
|
tti = config.kcpItem.tti
|
||||||
};
|
};
|
||||||
if (iobound.Equals("out"))
|
if (iobound == "out")
|
||||||
{
|
{
|
||||||
kcpSettings.uplinkCapacity = config.kcpItem.uplinkCapacity;
|
kcpSettings.uplinkCapacity = config.kcpItem.uplinkCapacity;
|
||||||
kcpSettings.downlinkCapacity = config.kcpItem.downlinkCapacity;
|
kcpSettings.downlinkCapacity = config.kcpItem.downlinkCapacity;
|
||||||
}
|
}
|
||||||
else if (iobound.Equals("in"))
|
else if (iobound == "in")
|
||||||
{
|
{
|
||||||
kcpSettings.uplinkCapacity = config.kcpItem.downlinkCapacity; ;
|
kcpSettings.uplinkCapacity = config.kcpItem.downlinkCapacity; ;
|
||||||
kcpSettings.downlinkCapacity = config.kcpItem.downlinkCapacity;
|
kcpSettings.downlinkCapacity = config.kcpItem.downlinkCapacity;
|
||||||
@@ -661,13 +631,6 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
streamSettings.wsSettings = wsSettings;
|
streamSettings.wsSettings = wsSettings;
|
||||||
|
|
||||||
//TlsSettings tlsSettings = new TlsSettings();
|
|
||||||
//tlsSettings.allowInsecure = config.allowInsecure();
|
|
||||||
//if (!string.IsNullOrWhiteSpace(host))
|
|
||||||
//{
|
|
||||||
// tlsSettings.serverName = host;
|
|
||||||
//}
|
|
||||||
//streamSettings.tlsSettings = tlsSettings;
|
|
||||||
break;
|
break;
|
||||||
//h2
|
//h2
|
||||||
case "h2":
|
case "h2":
|
||||||
@@ -681,9 +644,6 @@ namespace v2rayN.Handler
|
|||||||
|
|
||||||
streamSettings.httpSettings = httpSettings;
|
streamSettings.httpSettings = httpSettings;
|
||||||
|
|
||||||
//TlsSettings tlsSettings2 = new TlsSettings();
|
|
||||||
//tlsSettings2.allowInsecure = config.allowInsecure();
|
|
||||||
//streamSettings.tlsSettings = tlsSettings2;
|
|
||||||
break;
|
break;
|
||||||
//quic
|
//quic
|
||||||
case "quic":
|
case "quic":
|
||||||
@@ -719,12 +679,11 @@ namespace v2rayN.Handler
|
|||||||
permit_without_stream = config.grpcItem.permit_without_stream,
|
permit_without_stream = config.grpcItem.permit_without_stream,
|
||||||
initial_windows_size = config.grpcItem.initial_windows_size,
|
initial_windows_size = config.grpcItem.initial_windows_size,
|
||||||
};
|
};
|
||||||
|
|
||||||
streamSettings.grpcSettings = grpcSettings;
|
streamSettings.grpcSettings = grpcSettings;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
//tcp
|
//tcp
|
||||||
if (node.headerType.Equals(Global.TcpHeaderHttp))
|
if (node.headerType == Global.TcpHeaderHttp)
|
||||||
{
|
{
|
||||||
TcpSettings tcpSettings = new()
|
TcpSettings tcpSettings = new()
|
||||||
{
|
{
|
||||||
@@ -734,7 +693,7 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
if (iobound.Equals("out"))
|
if (iobound == "out")
|
||||||
{
|
{
|
||||||
//request Host
|
//request Host
|
||||||
string request = Utils.GetEmbedText(Global.v2raySampleHttprequestFileName);
|
string request = Utils.GetEmbedText(Global.v2raySampleHttprequestFileName);
|
||||||
@@ -753,7 +712,7 @@ namespace v2rayN.Handler
|
|||||||
request = request.Replace("$requestPath$", $"\"{pathHttp}\"");
|
request = request.Replace("$requestPath$", $"\"{pathHttp}\"");
|
||||||
tcpSettings.header.request = Utils.FromJson<object>(request);
|
tcpSettings.header.request = Utils.FromJson<object>(request);
|
||||||
}
|
}
|
||||||
else if (iobound.Equals("in"))
|
else if (iobound == "in")
|
||||||
{
|
{
|
||||||
//string response = Utils.GetEmbedText(Global.v2raySampleHttpresponseFileName);
|
//string response = Utils.GetEmbedText(Global.v2raySampleHttpresponseFileName);
|
||||||
//tcpSettings.header.response = Utils.FromJson<object>(response);
|
//tcpSettings.header.response = Utils.FromJson<object>(response);
|
||||||
@@ -789,7 +748,7 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
|
|
||||||
var obj = Utils.ParseJson(config.remoteDNS);
|
var obj = Utils.ParseJson(config.remoteDNS);
|
||||||
if (obj != null && obj.ContainsKey("servers"))
|
if (obj?.ContainsKey("servers") == true)
|
||||||
{
|
{
|
||||||
v2rayConfig.dns = obj;
|
v2rayConfig.dns = obj;
|
||||||
}
|
}
|
||||||
@@ -1171,30 +1130,24 @@ namespace v2rayN.Handler
|
|||||||
profileItem.remarks = $"import@{DateTime.Now.ToShortDateString()}";
|
profileItem.remarks = $"import@{DateTime.Now.ToShortDateString()}";
|
||||||
|
|
||||||
//tcp or kcp
|
//tcp or kcp
|
||||||
if (outbound.streamSettings != null
|
if (outbound.streamSettings?.network != null
|
||||||
&& outbound.streamSettings.network != null
|
|
||||||
&& !Utils.IsNullOrEmpty(outbound.streamSettings.network))
|
&& !Utils.IsNullOrEmpty(outbound.streamSettings.network))
|
||||||
{
|
{
|
||||||
profileItem.network = outbound.streamSettings.network;
|
profileItem.network = outbound.streamSettings.network;
|
||||||
}
|
}
|
||||||
|
|
||||||
//tcp http
|
//tcp http
|
||||||
if (outbound.streamSettings != null
|
if (outbound.streamSettings?.tcpSettings?.header != null
|
||||||
&& outbound.streamSettings.tcpSettings != null
|
|
||||||
&& outbound.streamSettings.tcpSettings.header != null
|
|
||||||
&& !Utils.IsNullOrEmpty(outbound.streamSettings.tcpSettings.header.type))
|
&& !Utils.IsNullOrEmpty(outbound.streamSettings.tcpSettings.header.type))
|
||||||
{
|
{
|
||||||
if (outbound.streamSettings.tcpSettings.header.type.Equals(Global.TcpHeaderHttp))
|
if (outbound.streamSettings.tcpSettings.header.type == Global.TcpHeaderHttp)
|
||||||
{
|
{
|
||||||
profileItem.headerType = outbound.streamSettings.tcpSettings.header.type;
|
profileItem.headerType = outbound.streamSettings.tcpSettings.header.type;
|
||||||
string request = Convert.ToString(outbound.streamSettings.tcpSettings.header.request);
|
string? request = Convert.ToString(outbound.streamSettings.tcpSettings.header.request);
|
||||||
if (!Utils.IsNullOrEmpty(request))
|
if (!Utils.IsNullOrEmpty(request))
|
||||||
{
|
{
|
||||||
V2rayTcpRequest v2rayTcpRequest = Utils.FromJson<V2rayTcpRequest>(request);
|
V2rayTcpRequest? v2rayTcpRequest = Utils.FromJson<V2rayTcpRequest>(request);
|
||||||
if (v2rayTcpRequest != null
|
if (v2rayTcpRequest?.headers?.Host?.Count > 0)
|
||||||
&& v2rayTcpRequest.headers != null
|
|
||||||
&& v2rayTcpRequest.headers.Host != null
|
|
||||||
&& v2rayTcpRequest.headers.Host.Count > 0)
|
|
||||||
{
|
{
|
||||||
profileItem.requestHost = v2rayTcpRequest.headers.Host[0];
|
profileItem.requestHost = v2rayTcpRequest.headers.Host[0];
|
||||||
}
|
}
|
||||||
@@ -1202,17 +1155,14 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
//kcp
|
//kcp
|
||||||
if (outbound.streamSettings != null
|
if (outbound?.streamSettings?.kcpSettings?.header != null
|
||||||
&& outbound.streamSettings.kcpSettings != null
|
|
||||||
&& outbound.streamSettings.kcpSettings.header != null
|
|
||||||
&& !Utils.IsNullOrEmpty(outbound.streamSettings.kcpSettings.header.type))
|
&& !Utils.IsNullOrEmpty(outbound.streamSettings.kcpSettings.header.type))
|
||||||
{
|
{
|
||||||
profileItem.headerType = outbound.streamSettings.kcpSettings.header.type;
|
profileItem.headerType = outbound.streamSettings.kcpSettings.header.type;
|
||||||
}
|
}
|
||||||
|
|
||||||
//ws
|
//ws
|
||||||
if (outbound.streamSettings != null
|
if (outbound?.streamSettings?.wsSettings != null)
|
||||||
&& outbound.streamSettings.wsSettings != null)
|
|
||||||
{
|
{
|
||||||
if (!Utils.IsNullOrEmpty(outbound.streamSettings.wsSettings.path))
|
if (!Utils.IsNullOrEmpty(outbound.streamSettings.wsSettings.path))
|
||||||
{
|
{
|
||||||
@@ -1226,24 +1176,20 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
|
|
||||||
//h2
|
//h2
|
||||||
if (outbound.streamSettings != null
|
if (outbound?.streamSettings?.httpSettings != null)
|
||||||
&& outbound.streamSettings.httpSettings != null)
|
|
||||||
{
|
{
|
||||||
if (!Utils.IsNullOrEmpty(outbound.streamSettings.httpSettings.path))
|
if (!Utils.IsNullOrEmpty(outbound.streamSettings.httpSettings.path))
|
||||||
{
|
{
|
||||||
profileItem.path = outbound.streamSettings.httpSettings.path;
|
profileItem.path = outbound.streamSettings.httpSettings.path;
|
||||||
}
|
}
|
||||||
if (outbound.streamSettings.httpSettings.host != null
|
if (outbound.streamSettings.httpSettings.host?.Count > 0)
|
||||||
&& outbound.streamSettings.httpSettings.host.Count > 0)
|
|
||||||
{
|
{
|
||||||
profileItem.requestHost = Utils.List2String(outbound.streamSettings.httpSettings.host);
|
profileItem.requestHost = Utils.List2String(outbound.streamSettings.httpSettings.host);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//tls
|
//tls
|
||||||
if (outbound.streamSettings != null
|
if (outbound?.streamSettings?.security == Global.StreamSecurity)
|
||||||
&& outbound.streamSettings.security != null
|
|
||||||
&& outbound.streamSettings.security == Global.StreamSecurity)
|
|
||||||
{
|
{
|
||||||
profileItem.streamSecurity = Global.StreamSecurity;
|
profileItem.streamSecurity = Global.StreamSecurity;
|
||||||
}
|
}
|
||||||
@@ -1265,7 +1211,7 @@ namespace v2rayN.Handler
|
|||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
string result = Utils.LoadResource(fileName);
|
string? result = Utils.LoadResource(fileName);
|
||||||
if (Utils.IsNullOrEmpty(result))
|
if (Utils.IsNullOrEmpty(result))
|
||||||
{
|
{
|
||||||
msg = ResUI.FailedReadConfiguration;
|
msg = ResUI.FailedReadConfiguration;
|
||||||
@@ -1309,30 +1255,24 @@ namespace v2rayN.Handler
|
|||||||
profileItem.remarks = $"import@{DateTime.Now.ToShortDateString()}";
|
profileItem.remarks = $"import@{DateTime.Now.ToShortDateString()}";
|
||||||
|
|
||||||
//tcp or kcp
|
//tcp or kcp
|
||||||
if (inbound.streamSettings != null
|
if (inbound.streamSettings?.network != null
|
||||||
&& inbound.streamSettings.network != null
|
|
||||||
&& !Utils.IsNullOrEmpty(inbound.streamSettings.network))
|
&& !Utils.IsNullOrEmpty(inbound.streamSettings.network))
|
||||||
{
|
{
|
||||||
profileItem.network = inbound.streamSettings.network;
|
profileItem.network = inbound.streamSettings.network;
|
||||||
}
|
}
|
||||||
|
|
||||||
//tcp http
|
//tcp http
|
||||||
if (inbound.streamSettings != null
|
if (inbound.streamSettings?.tcpSettings?.header != null
|
||||||
&& inbound.streamSettings.tcpSettings != null
|
|
||||||
&& inbound.streamSettings.tcpSettings.header != null
|
|
||||||
&& !Utils.IsNullOrEmpty(inbound.streamSettings.tcpSettings.header.type))
|
&& !Utils.IsNullOrEmpty(inbound.streamSettings.tcpSettings.header.type))
|
||||||
{
|
{
|
||||||
if (inbound.streamSettings.tcpSettings.header.type.Equals(Global.TcpHeaderHttp))
|
if (inbound.streamSettings.tcpSettings.header.type == Global.TcpHeaderHttp)
|
||||||
{
|
{
|
||||||
profileItem.headerType = inbound.streamSettings.tcpSettings.header.type;
|
profileItem.headerType = inbound.streamSettings.tcpSettings.header.type;
|
||||||
string request = Convert.ToString(inbound.streamSettings.tcpSettings.header.request);
|
string? request = Convert.ToString(inbound.streamSettings.tcpSettings.header.request);
|
||||||
if (!Utils.IsNullOrEmpty(request))
|
if (!Utils.IsNullOrEmpty(request))
|
||||||
{
|
{
|
||||||
V2rayTcpRequest v2rayTcpRequest = Utils.FromJson<V2rayTcpRequest>(request);
|
V2rayTcpRequest? v2rayTcpRequest = Utils.FromJson<V2rayTcpRequest>(request);
|
||||||
if (v2rayTcpRequest != null
|
if (v2rayTcpRequest?.headers?.Host?.Count > 0)
|
||||||
&& v2rayTcpRequest.headers != null
|
|
||||||
&& v2rayTcpRequest.headers.Host != null
|
|
||||||
&& v2rayTcpRequest.headers.Host.Count > 0)
|
|
||||||
{
|
{
|
||||||
profileItem.requestHost = v2rayTcpRequest.headers.Host[0];
|
profileItem.requestHost = v2rayTcpRequest.headers.Host[0];
|
||||||
}
|
}
|
||||||
@@ -1349,8 +1289,7 @@ namespace v2rayN.Handler
|
|||||||
//}
|
//}
|
||||||
|
|
||||||
//ws
|
//ws
|
||||||
if (inbound.streamSettings != null
|
if (inbound.streamSettings?.wsSettings != null)
|
||||||
&& inbound.streamSettings.wsSettings != null)
|
|
||||||
{
|
{
|
||||||
if (!Utils.IsNullOrEmpty(inbound.streamSettings.wsSettings.path))
|
if (!Utils.IsNullOrEmpty(inbound.streamSettings.wsSettings.path))
|
||||||
{
|
{
|
||||||
@@ -1364,24 +1303,20 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
|
|
||||||
//h2
|
//h2
|
||||||
if (inbound.streamSettings != null
|
if (inbound.streamSettings?.httpSettings != null)
|
||||||
&& inbound.streamSettings.httpSettings != null)
|
|
||||||
{
|
{
|
||||||
if (!Utils.IsNullOrEmpty(inbound.streamSettings.httpSettings.path))
|
if (!Utils.IsNullOrEmpty(inbound.streamSettings.httpSettings.path))
|
||||||
{
|
{
|
||||||
profileItem.path = inbound.streamSettings.httpSettings.path;
|
profileItem.path = inbound.streamSettings.httpSettings.path;
|
||||||
}
|
}
|
||||||
if (inbound.streamSettings.httpSettings.host != null
|
if (inbound.streamSettings.httpSettings.host?.Count > 0)
|
||||||
&& inbound.streamSettings.httpSettings.host.Count > 0)
|
|
||||||
{
|
{
|
||||||
profileItem.requestHost = Utils.List2String(inbound.streamSettings.httpSettings.host);
|
profileItem.requestHost = Utils.List2String(inbound.streamSettings.httpSettings.host);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
//tls
|
//tls
|
||||||
if (inbound.streamSettings != null
|
if (inbound.streamSettings?.security == Global.StreamSecurity)
|
||||||
&& inbound.streamSettings.security != null
|
|
||||||
&& inbound.streamSettings.security == Global.StreamSecurity)
|
|
||||||
{
|
{
|
||||||
profileItem.streamSecurity = Global.StreamSecurity;
|
profileItem.streamSecurity = Global.StreamSecurity;
|
||||||
}
|
}
|
||||||
@@ -1455,7 +1390,7 @@ namespace v2rayN.Handler
|
|||||||
Utils.SaveLog(ex.Message, ex);
|
Utils.SaveLog(ex.Message, ex);
|
||||||
}
|
}
|
||||||
|
|
||||||
log(configCopy, ref v2rayConfig, false);
|
log(configCopy, ref v2rayConfig, true);
|
||||||
//routing(config, ref v2rayConfig);
|
//routing(config, ref v2rayConfig);
|
||||||
//dns(configCopy, ref v2rayConfig);
|
//dns(configCopy, ref v2rayConfig);
|
||||||
|
|
||||||
@@ -1486,11 +1421,11 @@ namespace v2rayN.Handler
|
|||||||
var port = httpPort;
|
var port = httpPort;
|
||||||
for (int k = httpPort; k < Global.MaxPort; k++)
|
for (int k = httpPort; k < Global.MaxPort; k++)
|
||||||
{
|
{
|
||||||
if (lstIpEndPoints != null && lstIpEndPoints.FindIndex(_it => _it.Port == k) >= 0)
|
if (lstIpEndPoints?.FindIndex(_it => _it.Port == k) >= 0)
|
||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if (lstTcpConns != null && lstTcpConns.FindIndex(_it => _it.LocalEndPoint.Port == k) >= 0)
|
if (lstTcpConns?.FindIndex(_it => _it.LocalEndPoint.Port == k) >= 0)
|
||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@@ -1501,7 +1436,7 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
|
|
||||||
//Port In Used
|
//Port In Used
|
||||||
if (lstIpEndPoints != null && lstIpEndPoints.FindIndex(_it => _it.Port == port) >= 0)
|
if (lstIpEndPoints?.FindIndex(_it => _it.Port == port) >= 0)
|
||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -27,46 +27,43 @@ namespace v2rayN.Handler
|
|||||||
|
|
||||||
public void LoadCore(Config config)
|
public void LoadCore(Config config)
|
||||||
{
|
{
|
||||||
if (Global.reloadCore)
|
var node = ConfigHandler.GetDefaultServer(ref config);
|
||||||
|
if (node == null)
|
||||||
{
|
{
|
||||||
var node = ConfigHandler.GetDefaultServer(ref config);
|
ShowMsg(false, ResUI.CheckServerSettings);
|
||||||
if (node == null)
|
return;
|
||||||
{
|
}
|
||||||
ShowMsg(false, ResUI.CheckServerSettings);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (SetCore(config, node) != 0)
|
if (SetCore(config, node) != 0)
|
||||||
{
|
{
|
||||||
ShowMsg(false, ResUI.CheckServerSettings);
|
ShowMsg(false, ResUI.CheckServerSettings);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
string fileName = Utils.GetConfigPath(_coreCConfigRes);
|
string fileName = Utils.GetConfigPath(_coreCConfigRes);
|
||||||
if (CoreConfigHandler.GenerateClientConfig(node, fileName, out string msg, out string content) != 0)
|
if (CoreConfigHandler.GenerateClientConfig(node, fileName, out string msg, out string content) != 0)
|
||||||
{
|
{
|
||||||
ShowMsg(false, msg);
|
ShowMsg(false, msg);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
ShowMsg(false, msg);
|
ShowMsg(false, msg);
|
||||||
ShowMsg(true, $"{node.GetSummary()}");
|
ShowMsg(true, $"{node.GetSummary()}");
|
||||||
CoreStop();
|
CoreStop();
|
||||||
CoreStart(node);
|
CoreStart(node);
|
||||||
}
|
}
|
||||||
|
|
||||||
//start a socks service
|
//start a socks service
|
||||||
if (_process != null && !_process.HasExited && node.configType == EConfigType.Custom && node.preSocksPort > 0)
|
if (_process != null && !_process.HasExited && node.configType == EConfigType.Custom && node.preSocksPort > 0)
|
||||||
|
{
|
||||||
|
var itemSocks = new ProfileItem()
|
||||||
{
|
{
|
||||||
var itemSocks = new ProfileItem()
|
configType = EConfigType.Socks,
|
||||||
{
|
address = Global.Loopback,
|
||||||
configType = EConfigType.Socks,
|
port = node.preSocksPort
|
||||||
address = Global.Loopback,
|
};
|
||||||
port = node.preSocksPort
|
if (CoreConfigHandler.GenerateClientConfig(itemSocks, null, out string msg2, out string configStr) == 0)
|
||||||
};
|
{
|
||||||
if (CoreConfigHandler.GenerateClientConfig(itemSocks, null, out string msg2, out string configStr) == 0)
|
_processId = CoreStartViaString(configStr);
|
||||||
{
|
|
||||||
_processId = CoreStartViaString(configStr);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -159,6 +156,7 @@ namespace v2rayN.Handler
|
|||||||
if (Utils.IsNullOrEmpty(fileName))
|
if (Utils.IsNullOrEmpty(fileName))
|
||||||
{
|
{
|
||||||
string msg = string.Format(ResUI.NotFoundCore, Utils.GetBinPath("", coreInfo.coreType), string.Join(", ", coreInfo.coreExes.ToArray()), coreInfo.coreUrl);
|
string msg = string.Format(ResUI.NotFoundCore, Utils.GetBinPath("", coreInfo.coreType), string.Join(", ", coreInfo.coreExes.ToArray()), coreInfo.coreUrl);
|
||||||
|
Utils.SaveLog(msg);
|
||||||
ShowMsg(false, msg);
|
ShowMsg(false, msg);
|
||||||
}
|
}
|
||||||
return fileName;
|
return fileName;
|
||||||
@@ -216,6 +214,7 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
|
Utils.SaveLog(Utils.ToJson(node));
|
||||||
Utils.SaveLog(ex.Message, ex);
|
Utils.SaveLog(ex.Message, ex);
|
||||||
string msg = ex.Message;
|
string msg = ex.Message;
|
||||||
ShowMsg(true, msg);
|
ShowMsg(true, msg);
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ namespace v2rayN.Handler
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
class DownloadHandle
|
class DownloadHandle
|
||||||
{
|
{
|
||||||
public event EventHandler<ResultEventArgs> UpdateCompleted;
|
public event EventHandler<ResultEventArgs>? UpdateCompleted;
|
||||||
|
|
||||||
public event ErrorEventHandler? Error;
|
public event ErrorEventHandler? Error;
|
||||||
|
|
||||||
@@ -73,11 +73,7 @@ namespace v2rayN.Handler
|
|||||||
var progress = new Progress<double>();
|
var progress = new Progress<double>();
|
||||||
progress.ProgressChanged += (sender, value) =>
|
progress.ProgressChanged += (sender, value) =>
|
||||||
{
|
{
|
||||||
if (UpdateCompleted != null)
|
UpdateCompleted?.Invoke(this, new ResultEventArgs(value > 100, $"...{value}%"));
|
||||||
{
|
|
||||||
string msg = $"...{value}%";
|
|
||||||
UpdateCompleted(this, new ResultEventArgs(value > 100, msg));
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
var webProxy = GetWebProxy(blProxy);
|
var webProxy = GetWebProxy(blProxy);
|
||||||
@@ -110,9 +106,9 @@ namespace v2rayN.Handler
|
|||||||
HttpClient client = new(webRequestHandler);
|
HttpClient client = new(webRequestHandler);
|
||||||
|
|
||||||
HttpResponseMessage response = await client.GetAsync(url);
|
HttpResponseMessage response = await client.GetAsync(url);
|
||||||
if (response.StatusCode.ToString() == "Redirect")
|
if (response.StatusCode == HttpStatusCode.Redirect && response.Headers.Location is not null)
|
||||||
{
|
{
|
||||||
return response.Headers.Location?.ToString();
|
return response.Headers.Location.ToString();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -192,14 +188,16 @@ namespace v2rayN.Handler
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
Utils.SetSecurityProtocol(LazyConfig.Instance.GetConfig().guiItem.enableSecurityProtocolTls13);
|
Utils.SetSecurityProtocol(LazyConfig.Instance.GetConfig().guiItem.enableSecurityProtocolTls13);
|
||||||
|
var webProxy = GetWebProxy(blProxy);
|
||||||
var client = new HttpClient(new SocketsHttpHandler()
|
var client = new HttpClient(new SocketsHttpHandler()
|
||||||
{
|
{
|
||||||
Proxy = GetWebProxy(blProxy)
|
Proxy = webProxy,
|
||||||
|
UseProxy = webProxy != null
|
||||||
});
|
});
|
||||||
|
|
||||||
if (Utils.IsNullOrEmpty(userAgent))
|
if (Utils.IsNullOrEmpty(userAgent))
|
||||||
{
|
{
|
||||||
userAgent = $"{Utils.GetVersion(false)}";
|
userAgent = Utils.GetVersion(false);
|
||||||
}
|
}
|
||||||
client.DefaultRequestHeaders.UserAgent.TryParseAdd(userAgent);
|
client.DefaultRequestHeaders.UserAgent.TryParseAdd(userAgent);
|
||||||
|
|
||||||
@@ -213,7 +211,7 @@ namespace v2rayN.Handler
|
|||||||
var cts = new CancellationTokenSource();
|
var cts = new CancellationTokenSource();
|
||||||
cts.CancelAfter(1000 * 30);
|
cts.CancelAfter(1000 * 30);
|
||||||
|
|
||||||
var result = await HttpClientHelper.GetInstance().GetAsync(client, url, cts.Token);
|
var result = await HttpClientHelper.Instance.GetAsync(client, url, cts.Token);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
@@ -242,7 +240,7 @@ namespace v2rayN.Handler
|
|||||||
|
|
||||||
if (Utils.IsNullOrEmpty(userAgent))
|
if (Utils.IsNullOrEmpty(userAgent))
|
||||||
{
|
{
|
||||||
userAgent = $"{Utils.GetVersion(false)}";
|
userAgent = Utils.GetVersion(false);
|
||||||
}
|
}
|
||||||
var result = await DownloaderHelper.Instance.DownloadStringAsync(webProxy, url, userAgent, 30);
|
var result = await DownloaderHelper.Instance.DownloadStringAsync(webProxy, url, userAgent, 30);
|
||||||
return result;
|
return result;
|
||||||
@@ -260,7 +258,7 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public int RunAvailabilityCheck(WebProxy? webProxy)
|
public int RunAvailabilityCheck(IWebProxy? webProxy)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@@ -290,7 +288,7 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public string GetRealPingTime(string url, WebProxy webProxy, int downloadTimeout, out int responseTime)
|
public string GetRealPingTime(string url, IWebProxy? webProxy, int downloadTimeout, out int responseTime)
|
||||||
{
|
{
|
||||||
string msg = string.Empty;
|
string msg = string.Empty;
|
||||||
responseTime = -1;
|
responseTime = -1;
|
||||||
@@ -300,18 +298,14 @@ namespace v2rayN.Handler
|
|||||||
myHttpWebRequest.Timeout = downloadTimeout * 1000;
|
myHttpWebRequest.Timeout = downloadTimeout * 1000;
|
||||||
myHttpWebRequest.Proxy = webProxy;
|
myHttpWebRequest.Proxy = webProxy;
|
||||||
|
|
||||||
Stopwatch timer = new();
|
Stopwatch timer = Stopwatch.StartNew();
|
||||||
timer.Start();
|
|
||||||
|
|
||||||
HttpWebResponse myHttpWebResponse = (HttpWebResponse)myHttpWebRequest.GetResponse();
|
using HttpWebResponse myHttpWebResponse = (HttpWebResponse)myHttpWebRequest.GetResponse();
|
||||||
if (myHttpWebResponse.StatusCode is not HttpStatusCode.OK and not HttpStatusCode.NoContent)
|
if (myHttpWebResponse.StatusCode is not HttpStatusCode.OK and not HttpStatusCode.NoContent)
|
||||||
{
|
{
|
||||||
msg = myHttpWebResponse.StatusDescription;
|
msg = myHttpWebResponse.StatusDescription;
|
||||||
}
|
}
|
||||||
timer.Stop();
|
|
||||||
responseTime = timer.Elapsed.Milliseconds;
|
responseTime = timer.Elapsed.Milliseconds;
|
||||||
|
|
||||||
myHttpWebResponse.Close();
|
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
@@ -338,25 +332,17 @@ namespace v2rayN.Handler
|
|||||||
|
|
||||||
private bool SocketCheck(string ip, int port)
|
private bool SocketCheck(string ip, int port)
|
||||||
{
|
{
|
||||||
Socket? sock = null;
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
IPAddress ipa = IPAddress.Parse(ip);
|
IPEndPoint point = new(IPAddress.Parse(ip), port);
|
||||||
IPEndPoint point = new(ipa, port);
|
using Socket? sock = new(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
|
||||||
sock = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
|
|
||||||
sock.Connect(point);
|
sock.Connect(point);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
catch { }
|
catch (Exception)
|
||||||
finally
|
|
||||||
{
|
{
|
||||||
if (sock != null)
|
return false;
|
||||||
{
|
|
||||||
sock.Close();
|
|
||||||
sock.Dispose();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
172
v2rayN/v2rayN/Handler/HotkeyHandler.cs
Normal file
172
v2rayN/v2rayN/Handler/HotkeyHandler.cs
Normal file
@@ -0,0 +1,172 @@
|
|||||||
|
using System.ComponentModel;
|
||||||
|
using System.Runtime.InteropServices;
|
||||||
|
using System.Text;
|
||||||
|
using System.Windows;
|
||||||
|
using System.Windows.Input;
|
||||||
|
using System.Windows.Interop;
|
||||||
|
using v2rayN.Mode;
|
||||||
|
using v2rayN.Resx;
|
||||||
|
|
||||||
|
namespace v2rayN.Handler
|
||||||
|
{
|
||||||
|
public sealed class HotkeyHandler
|
||||||
|
{
|
||||||
|
private static readonly Lazy<HotkeyHandler> _instance = new(() => new());
|
||||||
|
public static HotkeyHandler Instance = _instance.Value;
|
||||||
|
|
||||||
|
private const int WmHotkey = 0x0312;
|
||||||
|
private Config _config
|
||||||
|
{
|
||||||
|
get => LazyConfig.Instance.GetConfig();
|
||||||
|
}
|
||||||
|
private Dictionary<int, List<EGlobalHotkey>> _hotkeyTriggerDic;
|
||||||
|
|
||||||
|
public bool IsPause { get; set; } = false;
|
||||||
|
public event Action<bool, string>? UpdateViewEvent;
|
||||||
|
public event Action<EGlobalHotkey>? HotkeyTriggerEvent;
|
||||||
|
public HotkeyHandler()
|
||||||
|
{
|
||||||
|
_hotkeyTriggerDic = new();
|
||||||
|
ComponentDispatcher.ThreadPreprocessMessage += OnThreadPreProcessMessage;
|
||||||
|
Init();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void Init()
|
||||||
|
{
|
||||||
|
_hotkeyTriggerDic.Clear();
|
||||||
|
if (_config.globalHotkeys == null) return;
|
||||||
|
foreach (var item in _config.globalHotkeys)
|
||||||
|
{
|
||||||
|
if (item.KeyCode != null && item.KeyCode != Key.None)
|
||||||
|
{
|
||||||
|
int key = KeyInterop.VirtualKeyFromKey((Key)item.KeyCode);
|
||||||
|
KeyModifiers modifiers = KeyModifiers.None;
|
||||||
|
if (item.Control) modifiers |= KeyModifiers.Ctrl;
|
||||||
|
if (item.Shift) modifiers |= KeyModifiers.Shift;
|
||||||
|
if (item.Alt) modifiers |= KeyModifiers.Alt;
|
||||||
|
key = (key << 16) | (int)modifiers;
|
||||||
|
if (!_hotkeyTriggerDic.ContainsKey(key))
|
||||||
|
{
|
||||||
|
_hotkeyTriggerDic.Add(key, new() { item.eGlobalHotkey });
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (!_hotkeyTriggerDic[key].Contains(item.eGlobalHotkey))
|
||||||
|
_hotkeyTriggerDic[key].Add(item.eGlobalHotkey);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public void Load()
|
||||||
|
{
|
||||||
|
foreach (var _hotkeyCode in _hotkeyTriggerDic.Keys)
|
||||||
|
{
|
||||||
|
var hotkeyInfo = GetHotkeyInfo(_hotkeyCode);
|
||||||
|
bool isSuccess = false;
|
||||||
|
string msg;
|
||||||
|
|
||||||
|
Application.Current.Dispatcher.Invoke(() =>
|
||||||
|
{
|
||||||
|
isSuccess = RegisterHotKey(IntPtr.Zero, _hotkeyCode, hotkeyInfo.fsModifiers, hotkeyInfo.vKey);
|
||||||
|
});
|
||||||
|
foreach (var name in hotkeyInfo.Names)
|
||||||
|
{
|
||||||
|
if (isSuccess)
|
||||||
|
{
|
||||||
|
msg = string.Format(ResUI.RegisterGlobalHotkeySuccessfully, $"{name}({hotkeyInfo.hotkeyStr})");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
var errInfo = new Win32Exception(Marshal.GetLastWin32Error()).Message;
|
||||||
|
msg = string.Format(ResUI.RegisterGlobalHotkeyFailed, $"{name}({hotkeyInfo.hotkeyStr})", errInfo);
|
||||||
|
}
|
||||||
|
UpdateViewEvent?.Invoke(false, msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void ReLoad()
|
||||||
|
{
|
||||||
|
foreach (var hotkey in _hotkeyTriggerDic.Keys)
|
||||||
|
{
|
||||||
|
Application.Current.Dispatcher.Invoke(() =>
|
||||||
|
{
|
||||||
|
UnregisterHotKey(IntPtr.Zero, hotkey);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
Init();
|
||||||
|
Load();
|
||||||
|
}
|
||||||
|
private (int fsModifiers, int vKey, string hotkeyStr, List<string> Names) GetHotkeyInfo(int hotkeycode)
|
||||||
|
{
|
||||||
|
var _fsModifiers = hotkeycode & 0xffff;
|
||||||
|
var _vkey = (hotkeycode >> 16) & 0xffff;
|
||||||
|
var _hotkeyStr = new StringBuilder();
|
||||||
|
var _names = new List<string>();
|
||||||
|
|
||||||
|
var mdif = (KeyModifiers)_fsModifiers;
|
||||||
|
var key = KeyInterop.KeyFromVirtualKey(_vkey);
|
||||||
|
if ((mdif | KeyModifiers.Ctrl) == KeyModifiers.Ctrl) _hotkeyStr.Append($"{KeyModifiers.Ctrl}+");
|
||||||
|
if ((mdif | KeyModifiers.Alt) == KeyModifiers.Alt) _hotkeyStr.Append($"{KeyModifiers.Alt}+");
|
||||||
|
if ((mdif | KeyModifiers.Shift) == KeyModifiers.Shift) _hotkeyStr.Append($"{KeyModifiers.Shift}+");
|
||||||
|
_hotkeyStr.Append(key.ToString());
|
||||||
|
|
||||||
|
foreach (var name in _hotkeyTriggerDic[hotkeycode])
|
||||||
|
{
|
||||||
|
_names.Add(name.ToString());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
return (_fsModifiers, _vkey, _hotkeyStr.ToString(), _names);
|
||||||
|
}
|
||||||
|
private void OnThreadPreProcessMessage(ref MSG msg, ref bool handled)
|
||||||
|
{
|
||||||
|
if (msg.message != WmHotkey || !_hotkeyTriggerDic.ContainsKey((int)msg.lParam))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
handled = true;
|
||||||
|
var _hotKeyCode = (int)msg.lParam;
|
||||||
|
if (IsPause)
|
||||||
|
{
|
||||||
|
Application.Current.Dispatcher.Invoke(() =>
|
||||||
|
{
|
||||||
|
UIElement? element = Keyboard.FocusedElement as UIElement;
|
||||||
|
if (element != null)
|
||||||
|
{
|
||||||
|
var _keyEventArgs = new KeyEventArgs(Keyboard.PrimaryDevice,
|
||||||
|
PresentationSource.FromVisual(element), 0,
|
||||||
|
KeyInterop.KeyFromVirtualKey(GetHotkeyInfo(_hotKeyCode).vKey))
|
||||||
|
{
|
||||||
|
RoutedEvent = UIElement.KeyDownEvent
|
||||||
|
};
|
||||||
|
element.RaiseEvent(_keyEventArgs);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
foreach (var keyEvent in _hotkeyTriggerDic[(int)msg.lParam])
|
||||||
|
{
|
||||||
|
HotkeyTriggerEvent?.Invoke(keyEvent);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
[DllImport("user32.dll", SetLastError = true)]
|
||||||
|
private static extern bool RegisterHotKey(IntPtr hWnd, int id, int fsModifiers, int vlc);
|
||||||
|
|
||||||
|
[DllImport("user32.dll", SetLastError = true)]
|
||||||
|
private static extern bool UnregisterHotKey(IntPtr hWnd, int id);
|
||||||
|
[Flags]
|
||||||
|
private enum KeyModifiers
|
||||||
|
{
|
||||||
|
None = 0x0000,
|
||||||
|
Alt = 0x0001,
|
||||||
|
Ctrl = 0x0002,
|
||||||
|
Shift = 0x0004,
|
||||||
|
Win = 0x0008,
|
||||||
|
NoRepeat = 0x4000
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -108,11 +108,11 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
if (!Utils.IsNullOrEmpty(filter))
|
if (!Utils.IsNullOrEmpty(filter))
|
||||||
{
|
{
|
||||||
if (filter.Contains("'"))
|
if (filter.Contains('\''))
|
||||||
{
|
{
|
||||||
filter = filter.Replace("'", "");
|
filter = filter.Replace("'", "");
|
||||||
}
|
}
|
||||||
sql += $" and a.remarks like '%{filter}%'";
|
sql += String.Format(" and (a.remarks like '%{0}%' or a.address like '%{0}%') ", filter);
|
||||||
}
|
}
|
||||||
|
|
||||||
return SqliteHelper.Instance.Query<ProfileItemModel>(sql).ToList();
|
return SqliteHelper.Instance.Query<ProfileItemModel>(sql).ToList();
|
||||||
@@ -156,7 +156,7 @@ namespace v2rayN.Handler
|
|||||||
|
|
||||||
public ECoreType GetCoreType(ProfileItem profileItem, EConfigType eConfigType)
|
public ECoreType GetCoreType(ProfileItem profileItem, EConfigType eConfigType)
|
||||||
{
|
{
|
||||||
if (profileItem != null && profileItem.coreType != null)
|
if (profileItem?.coreType != null)
|
||||||
{
|
{
|
||||||
return (ECoreType)profileItem.coreType;
|
return (ECoreType)profileItem.coreType;
|
||||||
}
|
}
|
||||||
@@ -179,7 +179,7 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
InitCoreInfo();
|
InitCoreInfo();
|
||||||
}
|
}
|
||||||
return coreInfos.Where(t => t.coreType == coreType).FirstOrDefault();
|
return coreInfos!.FirstOrDefault(t => t.coreType == coreType);
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<CoreInfo>? GetCoreInfos()
|
public List<CoreInfo>? GetCoreInfos()
|
||||||
|
|||||||
@@ -1,9 +1,7 @@
|
|||||||
using NHotkey;
|
|
||||||
using NHotkey.Wpf;
|
|
||||||
using System.Drawing;
|
using System.Drawing;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Windows.Forms;
|
using System.Windows.Forms;
|
||||||
using System.Windows.Input;
|
|
||||||
using System.Windows.Media.Imaging;
|
using System.Windows.Media.Imaging;
|
||||||
using v2rayN.Mode;
|
using v2rayN.Mode;
|
||||||
using v2rayN.Resx;
|
using v2rayN.Resx;
|
||||||
@@ -343,48 +341,11 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void RegisterGlobalHotkey(Config config, EventHandler<HotkeyEventArgs> handler, Action<bool, string> update)
|
public void RegisterGlobalHotkey(Config config, Action<EGlobalHotkey> handler, Action<bool, string> update)
|
||||||
{
|
{
|
||||||
if (config.globalHotkeys == null)
|
HotkeyHandler.Instance.UpdateViewEvent += update;
|
||||||
{
|
HotkeyHandler.Instance.HotkeyTriggerEvent += handler;
|
||||||
return;
|
HotkeyHandler.Instance.Load();
|
||||||
}
|
|
||||||
|
|
||||||
foreach (var item in config.globalHotkeys)
|
|
||||||
{
|
|
||||||
if (item.KeyCode == null)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
var modifiers = ModifierKeys.None;
|
|
||||||
if (item.Control)
|
|
||||||
{
|
|
||||||
modifiers |= ModifierKeys.Control;
|
|
||||||
}
|
|
||||||
if (item.Alt)
|
|
||||||
{
|
|
||||||
modifiers |= ModifierKeys.Alt;
|
|
||||||
}
|
|
||||||
if (item.Shift)
|
|
||||||
{
|
|
||||||
modifiers |= ModifierKeys.Shift;
|
|
||||||
}
|
|
||||||
|
|
||||||
var gesture = new KeyGesture(KeyInterop.KeyFromVirtualKey((int)item.KeyCode), modifiers);
|
|
||||||
try
|
|
||||||
{
|
|
||||||
HotkeyManager.Current.AddOrReplace(((int)item.eGlobalHotkey).ToString(), gesture, handler);
|
|
||||||
var msg = string.Format(ResUI.RegisterGlobalHotkeySuccessfully, $"{item.eGlobalHotkey}");
|
|
||||||
update(false, msg);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
var msg = string.Format(ResUI.RegisterGlobalHotkeyFailed, $"{item.eGlobalHotkey}", ex.Message);
|
|
||||||
update(false, msg);
|
|
||||||
Utils.SaveLog(msg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,6 @@
|
|||||||
using v2rayN.Base;
|
using System.Collections.Concurrent;
|
||||||
|
using System.Reactive.Linq;
|
||||||
|
using v2rayN.Base;
|
||||||
using v2rayN.Mode;
|
using v2rayN.Mode;
|
||||||
|
|
||||||
namespace v2rayN.Handler
|
namespace v2rayN.Handler
|
||||||
@@ -6,8 +8,9 @@ namespace v2rayN.Handler
|
|||||||
class ProfileExHandler
|
class ProfileExHandler
|
||||||
{
|
{
|
||||||
private static readonly Lazy<ProfileExHandler> _instance = new(() => new());
|
private static readonly Lazy<ProfileExHandler> _instance = new(() => new());
|
||||||
private List<ProfileExItem> _lstProfileEx;
|
private ConcurrentBag<ProfileExItem> _lstProfileEx;
|
||||||
public List<ProfileExItem> ProfileExs => _lstProfileEx;
|
private Queue<string> _queIndexIds = new();
|
||||||
|
public ConcurrentBag<ProfileExItem> ProfileExs => _lstProfileEx;
|
||||||
public static ProfileExHandler Instance => _instance.Value;
|
public static ProfileExHandler Instance => _instance.Value;
|
||||||
|
|
||||||
public ProfileExHandler()
|
public ProfileExHandler()
|
||||||
@@ -19,7 +22,33 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
SqliteHelper.Instance.Execute($"delete from ProfileExItem where indexId not in ( select indexId from ProfileItem )");
|
SqliteHelper.Instance.Execute($"delete from ProfileExItem where indexId not in ( select indexId from ProfileItem )");
|
||||||
|
|
||||||
_lstProfileEx = SqliteHelper.Instance.Table<ProfileExItem>().ToList();
|
_lstProfileEx = new(SqliteHelper.Instance.Table<ProfileExItem>());
|
||||||
|
|
||||||
|
Task.Run(() =>
|
||||||
|
{
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
var cnt = _queIndexIds.Count;
|
||||||
|
for (int i = 0; i < cnt; i++)
|
||||||
|
{
|
||||||
|
var id = _queIndexIds.Dequeue();
|
||||||
|
var item = _lstProfileEx.FirstOrDefault(t => t.indexId == id);
|
||||||
|
if (item is not null)
|
||||||
|
{
|
||||||
|
SqliteHelper.Instance.Replace(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Thread.Sleep(1000 * 60);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private void IndexIdEnqueue(string indexId)
|
||||||
|
{
|
||||||
|
if (!Utils.IsNullOrEmpty(indexId) && !_queIndexIds.Contains(indexId))
|
||||||
|
{
|
||||||
|
_queIndexIds.Enqueue(indexId);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void AddProfileEx(string indexId, ref ProfileExItem profileEx)
|
private void AddProfileEx(string indexId, ref ProfileExItem profileEx)
|
||||||
@@ -32,7 +61,7 @@ namespace v2rayN.Handler
|
|||||||
sort = 0
|
sort = 0
|
||||||
};
|
};
|
||||||
_lstProfileEx.Add(profileEx);
|
_lstProfileEx.Add(profileEx);
|
||||||
//SqliteHelper.Instance.Replace(profileEx);
|
IndexIdEnqueue(indexId);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void ClearAll()
|
public void ClearAll()
|
||||||
@@ -45,11 +74,11 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
foreach (var item in _lstProfileEx)
|
//foreach (var item in _lstProfileEx)
|
||||||
{
|
//{
|
||||||
SqliteHelper.Instance.Replace(item);
|
// SqliteHelper.Instance.Replace(item);
|
||||||
}
|
//}
|
||||||
//SqliteHelper.Instance.UpdateAll(_lstProfileEx);
|
SqliteHelper.Instance.UpdateAll(_lstProfileEx);
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
@@ -57,7 +86,7 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public Task SetTestDelay(string indexId, string delayVal)
|
public void SetTestDelay(string indexId, string delayVal)
|
||||||
{
|
{
|
||||||
var profileEx = _lstProfileEx.FirstOrDefault(t => t.indexId == indexId);
|
var profileEx = _lstProfileEx.FirstOrDefault(t => t.indexId == indexId);
|
||||||
if (profileEx == null)
|
if (profileEx == null)
|
||||||
@@ -67,10 +96,10 @@ namespace v2rayN.Handler
|
|||||||
|
|
||||||
int.TryParse(delayVal, out int delay);
|
int.TryParse(delayVal, out int delay);
|
||||||
profileEx.delay = delay;
|
profileEx.delay = delay;
|
||||||
return Task.CompletedTask;
|
IndexIdEnqueue(indexId);
|
||||||
}
|
}
|
||||||
|
|
||||||
public Task SetTestSpeed(string indexId, string speedVal)
|
public void SetTestSpeed(string indexId, string speedVal)
|
||||||
{
|
{
|
||||||
var profileEx = _lstProfileEx.FirstOrDefault(t => t.indexId == indexId);
|
var profileEx = _lstProfileEx.FirstOrDefault(t => t.indexId == indexId);
|
||||||
if (profileEx == null)
|
if (profileEx == null)
|
||||||
@@ -80,7 +109,7 @@ namespace v2rayN.Handler
|
|||||||
|
|
||||||
decimal.TryParse(speedVal, out decimal speed);
|
decimal.TryParse(speedVal, out decimal speed);
|
||||||
profileEx.speed = speed;
|
profileEx.speed = speed;
|
||||||
return Task.CompletedTask;
|
IndexIdEnqueue(indexId);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void SetSort(string indexId, int sort)
|
public void SetSort(string indexId, int sort)
|
||||||
@@ -91,6 +120,7 @@ namespace v2rayN.Handler
|
|||||||
AddProfileEx(indexId, ref profileEx);
|
AddProfileEx(indexId, ref profileEx);
|
||||||
}
|
}
|
||||||
profileEx.sort = sort;
|
profileEx.sort = sort;
|
||||||
|
IndexIdEnqueue(indexId);
|
||||||
}
|
}
|
||||||
|
|
||||||
public int GetSort(string indexId)
|
public int GetSort(string indexId)
|
||||||
|
|||||||
@@ -197,6 +197,18 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
dicQuery.Add("fp", Utils.UrlEncode(item.fingerprint));
|
dicQuery.Add("fp", Utils.UrlEncode(item.fingerprint));
|
||||||
}
|
}
|
||||||
|
if (!Utils.IsNullOrEmpty(item.publicKey))
|
||||||
|
{
|
||||||
|
dicQuery.Add("pbk", Utils.UrlEncode(item.publicKey));
|
||||||
|
}
|
||||||
|
if (!Utils.IsNullOrEmpty(item.shortId))
|
||||||
|
{
|
||||||
|
dicQuery.Add("sid", Utils.UrlEncode(item.shortId));
|
||||||
|
}
|
||||||
|
if (!Utils.IsNullOrEmpty(item.spiderX))
|
||||||
|
{
|
||||||
|
dicQuery.Add("spx", Utils.UrlEncode(item.spiderX));
|
||||||
|
}
|
||||||
|
|
||||||
dicQuery.Add("type", !Utils.IsNullOrEmpty(item.network) ? item.network : "tcp");
|
dicQuery.Add("type", !Utils.IsNullOrEmpty(item.network) ? item.network : "tcp");
|
||||||
|
|
||||||
@@ -274,7 +286,7 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
msg = string.Empty;
|
msg = string.Empty;
|
||||||
|
|
||||||
ProfileItem profileItem = new();
|
ProfileItem profileItem = new();
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@@ -756,6 +768,10 @@ namespace v2rayN.Handler
|
|||||||
item.sni = query["sni"] ?? "";
|
item.sni = query["sni"] ?? "";
|
||||||
item.alpn = Utils.UrlDecode(query["alpn"] ?? "");
|
item.alpn = Utils.UrlDecode(query["alpn"] ?? "");
|
||||||
item.fingerprint = Utils.UrlDecode(query["fp"] ?? "");
|
item.fingerprint = Utils.UrlDecode(query["fp"] ?? "");
|
||||||
|
item.publicKey = Utils.UrlDecode(query["pbk"] ?? "");
|
||||||
|
item.shortId = Utils.UrlDecode(query["sid"] ?? "");
|
||||||
|
item.spiderX = Utils.UrlDecode(query["spx"] ?? "");
|
||||||
|
|
||||||
item.network = query["type"] ?? "tcp";
|
item.network = query["type"] ?? "tcp";
|
||||||
switch (item.network)
|
switch (item.network)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -55,12 +55,16 @@ namespace v2rayN.Handler
|
|||||||
case ESpeedActionType.Tcping:
|
case ESpeedActionType.Tcping:
|
||||||
case ESpeedActionType.Realping:
|
case ESpeedActionType.Realping:
|
||||||
UpdateFunc(it.indexId, ResUI.Speedtesting, "");
|
UpdateFunc(it.indexId, ResUI.Speedtesting, "");
|
||||||
|
ProfileExHandler.Instance.SetTestDelay(it.indexId, "0");
|
||||||
break;
|
break;
|
||||||
case ESpeedActionType.Speedtest:
|
case ESpeedActionType.Speedtest:
|
||||||
UpdateFunc(it.indexId, "", ResUI.Speedtesting);
|
UpdateFunc(it.indexId, "", ResUI.SpeedtestingWait);
|
||||||
|
ProfileExHandler.Instance.SetTestSpeed(it.indexId, "0");
|
||||||
break;
|
break;
|
||||||
case ESpeedActionType.Mixedtest:
|
case ESpeedActionType.Mixedtest:
|
||||||
UpdateFunc(it.indexId, ResUI.Speedtesting, ResUI.Speedtesting);
|
UpdateFunc(it.indexId, ResUI.Speedtesting, ResUI.SpeedtestingWait);
|
||||||
|
ProfileExHandler.Instance.SetTestDelay(it.indexId, "0");
|
||||||
|
ProfileExHandler.Instance.SetTestSpeed(it.indexId, "0");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -112,24 +116,24 @@ namespace v2rayN.Handler
|
|||||||
|
|
||||||
private void RunPing()
|
private void RunPing()
|
||||||
{
|
{
|
||||||
RunPingSub(async (ServerTestItem it) =>
|
RunPingSub((ServerTestItem it) =>
|
||||||
{
|
{
|
||||||
long time = Ping(it.address);
|
long time = Ping(it.address);
|
||||||
var output = FormatOut(time, Global.DelayUnit);
|
var output = FormatOut(time, Global.DelayUnit);
|
||||||
|
|
||||||
await ProfileExHandler.Instance.SetTestDelay(it.indexId, output);
|
ProfileExHandler.Instance.SetTestDelay(it.indexId, output);
|
||||||
UpdateFunc(it.indexId, output);
|
UpdateFunc(it.indexId, output);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void RunTcping()
|
private void RunTcping()
|
||||||
{
|
{
|
||||||
RunPingSub(async (ServerTestItem it) =>
|
RunPingSub((ServerTestItem it) =>
|
||||||
{
|
{
|
||||||
int time = GetTcpingTime(it.address, it.port);
|
int time = GetTcpingTime(it.address, it.port);
|
||||||
var output = FormatOut(time, Global.DelayUnit);
|
var output = FormatOut(time, Global.DelayUnit);
|
||||||
|
|
||||||
await ProfileExHandler.Instance.SetTestDelay(it.indexId, output);
|
ProfileExHandler.Instance.SetTestDelay(it.indexId, output);
|
||||||
UpdateFunc(it.indexId, output);
|
UpdateFunc(it.indexId, output);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -161,7 +165,7 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
tasks.Add(Task.Run(async () =>
|
tasks.Add(Task.Run(() =>
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@@ -169,7 +173,7 @@ namespace v2rayN.Handler
|
|||||||
WebProxy webProxy = new(Global.Loopback, it.port);
|
WebProxy webProxy = new(Global.Loopback, it.port);
|
||||||
string output = GetRealPingTime(downloadHandle, webProxy);
|
string output = GetRealPingTime(downloadHandle, webProxy);
|
||||||
|
|
||||||
await ProfileExHandler.Instance.SetTestDelay(it.indexId, output);
|
ProfileExHandler.Instance.SetTestDelay(it.indexId, output);
|
||||||
UpdateFunc(it.indexId, output);
|
UpdateFunc(it.indexId, output);
|
||||||
int.TryParse(output, out int delay);
|
int.TryParse(output, out int delay);
|
||||||
it.delay = delay;
|
it.delay = delay;
|
||||||
@@ -231,7 +235,8 @@ namespace v2rayN.Handler
|
|||||||
// UpdateFunc(it.indexId, "", ResUI.SpeedtestingSkip);
|
// UpdateFunc(it.indexId, "", ResUI.SpeedtestingSkip);
|
||||||
// continue;
|
// continue;
|
||||||
//}
|
//}
|
||||||
await ProfileExHandler.Instance.SetTestSpeed(it.indexId, "-1");
|
ProfileExHandler.Instance.SetTestSpeed(it.indexId, "-1");
|
||||||
|
UpdateFunc(it.indexId, "", ResUI.Speedtesting);
|
||||||
|
|
||||||
var item = LazyConfig.Instance.GetProfileItem(it.indexId);
|
var item = LazyConfig.Instance.GetProfileItem(it.indexId);
|
||||||
if (item is null) continue;
|
if (item is null) continue;
|
||||||
@@ -243,7 +248,7 @@ namespace v2rayN.Handler
|
|||||||
decimal.TryParse(msg, out decimal dec);
|
decimal.TryParse(msg, out decimal dec);
|
||||||
if (dec > 0)
|
if (dec > 0)
|
||||||
{
|
{
|
||||||
await ProfileExHandler.Instance.SetTestSpeed(it.indexId, msg);
|
ProfileExHandler.Instance.SetTestSpeed(it.indexId, msg);
|
||||||
}
|
}
|
||||||
UpdateFunc(it.indexId, "", msg);
|
UpdateFunc(it.indexId, "", msg);
|
||||||
});
|
});
|
||||||
@@ -282,7 +287,8 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
await ProfileExHandler.Instance.SetTestSpeed(it.indexId, "-1");
|
ProfileExHandler.Instance.SetTestSpeed(it.indexId, "-1");
|
||||||
|
UpdateFunc(it.indexId, "", ResUI.Speedtesting);
|
||||||
|
|
||||||
var item = LazyConfig.Instance.GetProfileItem(it.indexId);
|
var item = LazyConfig.Instance.GetProfileItem(it.indexId);
|
||||||
if (item is null) continue;
|
if (item is null) continue;
|
||||||
@@ -293,7 +299,7 @@ namespace v2rayN.Handler
|
|||||||
decimal.TryParse(msg, out decimal dec);
|
decimal.TryParse(msg, out decimal dec);
|
||||||
if (dec > 0)
|
if (dec > 0)
|
||||||
{
|
{
|
||||||
await ProfileExHandler.Instance.SetTestSpeed(it.indexId, msg);
|
ProfileExHandler.Instance.SetTestSpeed(it.indexId, msg);
|
||||||
}
|
}
|
||||||
UpdateFunc(it.indexId, "", msg);
|
UpdateFunc(it.indexId, "", msg);
|
||||||
});
|
});
|
||||||
@@ -319,7 +325,7 @@ namespace v2rayN.Handler
|
|||||||
await RunSpeedTestMulti();
|
await RunSpeedTestMulti();
|
||||||
}
|
}
|
||||||
|
|
||||||
public string GetRealPingTime(DownloadHandle downloadHandle, WebProxy webProxy)
|
public string GetRealPingTime(DownloadHandle downloadHandle, IWebProxy webProxy)
|
||||||
{
|
{
|
||||||
string status = downloadHandle.GetRealPingTime(_config.speedTestItem.speedPingTestUrl, webProxy, 10, out int responseTime);
|
string status = downloadHandle.GetRealPingTime(_config.speedTestItem.speedPingTestUrl, webProxy, 10, out int responseTime);
|
||||||
//string output = Utils.IsNullOrEmpty(status) ? FormatOut(responseTime, "ms") : status;
|
//string output = Utils.IsNullOrEmpty(status) ? FormatOut(responseTime, "ms") : status;
|
||||||
|
|||||||
@@ -162,7 +162,7 @@ namespace v2rayN.Handler
|
|||||||
todayDown = 0,
|
todayDown = 0,
|
||||||
dateNow = ticks
|
dateNow = ticks
|
||||||
};
|
};
|
||||||
_ = SqliteHelper.Instance.Replacesync(_serverStatItem);
|
SqliteHelper.Instance.Replace(_serverStatItem);
|
||||||
_lstServerStat.Add(_serverStatItem);
|
_lstServerStat.Add(_serverStatItem);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -50,7 +50,7 @@ namespace v2rayN.Handler
|
|||||||
{
|
{
|
||||||
var type = config.sysProxyType;
|
var type = config.sysProxyType;
|
||||||
|
|
||||||
if (forceDisable && type == ESysProxyType.ForcedChange)
|
if (forceDisable && type != ESysProxyType.Unchanged)
|
||||||
{
|
{
|
||||||
type = ESysProxyType.ForcedClear;
|
type = ESysProxyType.ForcedClear;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -45,7 +45,7 @@ namespace v2rayN.Base
|
|||||||
{
|
{
|
||||||
var socksPort = LazyConfig.Instance.GetLocalPort(Global.InboundSocks);
|
var socksPort = LazyConfig.Instance.GetLocalPort(Global.InboundSocks);
|
||||||
|
|
||||||
if (socksPort.Equals(_socksPort)
|
if (socksPort == _socksPort
|
||||||
&& _process != null
|
&& _process != null
|
||||||
&& !_process.HasExited)
|
&& !_process.HasExited)
|
||||||
{
|
{
|
||||||
@@ -103,7 +103,7 @@ namespace v2rayN.Base
|
|||||||
configStr = configStr.Replace("$stack$", $"{_config.tunModeItem.stack}");
|
configStr = configStr.Replace("$stack$", $"{_config.tunModeItem.stack}");
|
||||||
|
|
||||||
//logs
|
//logs
|
||||||
configStr = configStr.Replace("$log_disabled$", $"aaa{(!_config.tunModeItem.enabledLog).ToString().ToLower()}");
|
configStr = configStr.Replace("$log_disabled$", $"{(!_config.tunModeItem.enabledLog).ToString().ToLower()}");
|
||||||
if (_config.tunModeItem.showWindow)
|
if (_config.tunModeItem.showWindow)
|
||||||
{
|
{
|
||||||
configStr = configStr.Replace("$log_output$", $"");
|
configStr = configStr.Replace("$log_output$", $"");
|
||||||
@@ -118,6 +118,22 @@ namespace v2rayN.Base
|
|||||||
//port
|
//port
|
||||||
configStr = configStr.Replace("$socksPort$", $"{_socksPort}");
|
configStr = configStr.Replace("$socksPort$", $"{_socksPort}");
|
||||||
|
|
||||||
|
//dns
|
||||||
|
string dnsObject = String.Empty;
|
||||||
|
if (_config.tunModeItem.bypassMode)
|
||||||
|
{
|
||||||
|
dnsObject = _config.tunModeItem.directDNS;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
dnsObject = _config.tunModeItem.proxyDNS;
|
||||||
|
}
|
||||||
|
if (dnsObject.IsNullOrEmpty() || Utils.ParseJson(dnsObject)?.ContainsKey("servers") == false)
|
||||||
|
{
|
||||||
|
dnsObject = Utils.GetEmbedText(Global.TunSingboxDNSFileName);
|
||||||
|
}
|
||||||
|
configStr = configStr.Replace("$dns_object$", dnsObject);
|
||||||
|
|
||||||
//exe
|
//exe
|
||||||
List<string> lstDnsExe = new();
|
List<string> lstDnsExe = new();
|
||||||
List<string> lstDirectExe = new();
|
List<string> lstDirectExe = new();
|
||||||
|
|||||||
@@ -164,35 +164,31 @@ namespace v2rayN.Handler
|
|||||||
Task.Run(async () =>
|
Task.Run(async () =>
|
||||||
{
|
{
|
||||||
//Turn off system proxy
|
//Turn off system proxy
|
||||||
bool bSysProxyType = false;
|
//bool bSysProxyType = false;
|
||||||
if (!blProxy && config.sysProxyType == ESysProxyType.ForcedChange)
|
//if (!blProxy && config.sysProxyType == ESysProxyType.ForcedChange)
|
||||||
{
|
//{
|
||||||
bSysProxyType = true;
|
// bSysProxyType = true;
|
||||||
config.sysProxyType = ESysProxyType.ForcedClear;
|
// config.sysProxyType = ESysProxyType.ForcedClear;
|
||||||
SysProxyHandle.UpdateSysProxy(config, false);
|
// SysProxyHandle.UpdateSysProxy(config, false);
|
||||||
Thread.Sleep(3000);
|
// Thread.Sleep(3000);
|
||||||
}
|
//}
|
||||||
|
|
||||||
foreach (var item in subItem)
|
foreach (var item in subItem)
|
||||||
{
|
{
|
||||||
if (item.enabled == false)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (!Utils.IsNullOrEmpty(subId) && item.id != subId)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
string id = item.id.TrimEx();
|
string id = item.id.TrimEx();
|
||||||
string url = item.url.TrimEx();
|
string url = item.url.TrimEx();
|
||||||
string userAgent = item.userAgent.TrimEx();
|
string userAgent = item.userAgent.TrimEx();
|
||||||
string hashCode = $"{item.remarks}->";
|
string hashCode = $"{item.remarks}->";
|
||||||
if (Utils.IsNullOrEmpty(id) || Utils.IsNullOrEmpty(url))
|
if (Utils.IsNullOrEmpty(id) || Utils.IsNullOrEmpty(url) || (!Utils.IsNullOrEmpty(subId) && item.id != subId))
|
||||||
{
|
{
|
||||||
//_updateFunc(false, $"{hashCode}{ResUI.MsgNoValidSubscription}");
|
//_updateFunc(false, $"{hashCode}{ResUI.MsgNoValidSubscription}");
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
if (item.enabled == false)
|
||||||
|
{
|
||||||
|
_updateFunc(false, $"{hashCode}{ResUI.MsgSkipSubscriptionUpdate}");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
var downloadHandle = new DownloadHandle();
|
var downloadHandle = new DownloadHandle();
|
||||||
downloadHandle.Error += (sender2, args) =>
|
downloadHandle.Error += (sender2, args) =>
|
||||||
@@ -235,12 +231,12 @@ namespace v2rayN.Handler
|
|||||||
}
|
}
|
||||||
_updateFunc(false, "-------------------------------------------------------");
|
_updateFunc(false, "-------------------------------------------------------");
|
||||||
}
|
}
|
||||||
//restore system proxy
|
////restore system proxy
|
||||||
if (bSysProxyType)
|
//if (bSysProxyType)
|
||||||
{
|
//{
|
||||||
config.sysProxyType = ESysProxyType.ForcedChange;
|
// config.sysProxyType = ESysProxyType.ForcedChange;
|
||||||
SysProxyHandle.UpdateSysProxy(config, false);
|
// SysProxyHandle.UpdateSysProxy(config, false);
|
||||||
}
|
//}
|
||||||
_updateFunc(true, $"{ResUI.MsgUpdateSubscriptionEnd}");
|
_updateFunc(true, $"{ResUI.MsgUpdateSubscriptionEnd}");
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -9,6 +9,7 @@
|
|||||||
#region property
|
#region property
|
||||||
|
|
||||||
public string indexId { get; set; }
|
public string indexId { get; set; }
|
||||||
|
public string subIndexId { get; set; }
|
||||||
|
|
||||||
public string remoteDNS { get; set; }
|
public string remoteDNS { get; set; }
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
using System.Windows.Forms;
|
using System.Windows.Input;
|
||||||
|
|
||||||
namespace v2rayN.Mode
|
namespace v2rayN.Mode
|
||||||
{
|
{
|
||||||
[Serializable]
|
[Serializable]
|
||||||
@@ -104,6 +103,10 @@ namespace v2rayN.Mode
|
|||||||
public bool enableSecurityProtocolTls13 { get; set; }
|
public bool enableSecurityProtocolTls13 { get; set; }
|
||||||
|
|
||||||
public int trayMenuServersLimit { get; set; } = 20;
|
public int trayMenuServersLimit { get; set; } = 20;
|
||||||
|
|
||||||
|
public bool enableHWA { get; set; } = false;
|
||||||
|
|
||||||
|
public bool enableLog { get; set; } = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
[Serializable]
|
[Serializable]
|
||||||
@@ -123,7 +126,8 @@ namespace v2rayN.Mode
|
|||||||
public bool doubleClick2Activate { get; set; }
|
public bool doubleClick2Activate { get; set; }
|
||||||
public bool autoHideStartup { get; set; } = true;
|
public bool autoHideStartup { get; set; } = true;
|
||||||
public string mainMsgFilter { get; set; }
|
public string mainMsgFilter { get; set; }
|
||||||
public Dictionary<string, int> mainLvColWidth { get; set; }
|
public List<ColumnItem> mainColumnItem { get; set; }
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
[Serializable]
|
[Serializable]
|
||||||
@@ -143,7 +147,7 @@ namespace v2rayN.Mode
|
|||||||
|
|
||||||
public bool Shift { get; set; }
|
public bool Shift { get; set; }
|
||||||
|
|
||||||
public Keys? KeyCode { get; set; }
|
public Key? KeyCode { get; set; }
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -168,8 +172,10 @@ namespace v2rayN.Mode
|
|||||||
public bool bypassMode { get; set; } = true;
|
public bool bypassMode { get; set; } = true;
|
||||||
public List<string> directIP { get; set; }
|
public List<string> directIP { get; set; }
|
||||||
public List<string> directProcess { get; set; }
|
public List<string> directProcess { get; set; }
|
||||||
|
public string directDNS { get; set; }
|
||||||
public List<string> proxyIP { get; set; }
|
public List<string> proxyIP { get; set; }
|
||||||
public List<string> proxyProcess { get; set; }
|
public List<string> proxyProcess { get; set; }
|
||||||
|
public string proxyDNS { get; set; }
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -193,4 +199,12 @@ namespace v2rayN.Mode
|
|||||||
public string routingIndexId { get; set; }
|
public string routingIndexId { get; set; }
|
||||||
public bool enableRoutingAdvanced { get; set; }
|
public bool enableRoutingAdvanced { get; set; }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Serializable]
|
||||||
|
public class ColumnItem
|
||||||
|
{
|
||||||
|
public string Name { get; set; }
|
||||||
|
public int Width { get; set; }
|
||||||
|
public int Index { get; set; }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,8 +12,8 @@ namespace v2rayN.Mode
|
|||||||
network,
|
network,
|
||||||
streamSecurity,
|
streamSecurity,
|
||||||
subRemarks,
|
subRemarks,
|
||||||
delay,
|
delayVal,
|
||||||
speed,
|
speedVal,
|
||||||
|
|
||||||
todayDown,
|
todayDown,
|
||||||
todayUp,
|
todayUp,
|
||||||
|
|||||||
@@ -85,158 +85,100 @@ namespace v2rayN.Mode
|
|||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
[PrimaryKey]
|
[PrimaryKey]
|
||||||
public string indexId
|
public string indexId { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// config type(1=normal,2=custom)
|
/// config type(1=normal,2=custom)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public EConfigType configType
|
public EConfigType configType { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 版本(现在=2)
|
/// 版本(现在=2)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public int configVersion
|
public int configVersion { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 远程服务器地址
|
/// 远程服务器地址
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string address
|
public string address { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 远程服务器端口
|
/// 远程服务器端口
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public int port
|
public int port { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 远程服务器ID
|
/// 远程服务器ID
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string id
|
public string id { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 远程服务器额外ID
|
/// 远程服务器额外ID
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public int alterId
|
public int alterId { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 本地安全策略
|
/// 本地安全策略
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string security
|
public string security { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// tcp,kcp,ws,h2,quic
|
/// tcp,kcp,ws,h2,quic
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string network
|
public string network { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 备注或别名
|
/// 备注或别名
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string remarks
|
public string remarks { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 伪装类型
|
/// 伪装类型
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string headerType
|
public string headerType { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 伪装的域名
|
/// 伪装的域名
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string requestHost
|
public string requestHost { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// ws h2 path
|
/// ws h2 path
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string path
|
public string path { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 传输层安全
|
/// 传输层安全
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string streamSecurity
|
public string streamSecurity { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 是否允许不安全连接(用于客户端)
|
/// 是否允许不安全连接(用于客户端)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string allowInsecure
|
public string allowInsecure { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// SubItem id
|
/// SubItem id
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string subid
|
public string subid { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
public bool isSub { get; set; } = true;
|
public bool isSub { get; set; } = true;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// VLESS flow
|
/// VLESS flow
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string flow
|
public string flow { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// tls sni
|
/// tls sni
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string sni
|
public string sni { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// tls alpn
|
/// tls alpn
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string alpn { get; set; } = string.Empty;
|
public string alpn { get; set; } = string.Empty;
|
||||||
|
|
||||||
|
|
||||||
public ECoreType? coreType
|
public ECoreType? coreType { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int preSocksPort
|
public int preSocksPort { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
|
|
||||||
public string fingerprint { get; set; }
|
public string fingerprint { get; set; }
|
||||||
|
|
||||||
public bool displayLog { get; set; } = true;
|
public bool displayLog { get; set; } = true;
|
||||||
|
public string publicKey { get; set; }
|
||||||
|
public string shortId { get; set; }
|
||||||
|
public string spiderX { get; set; }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -389,9 +389,9 @@ namespace v2rayN.Mode
|
|||||||
public QuicSettings quicSettings { get; set; }
|
public QuicSettings quicSettings { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// VLESS xtls
|
/// VLESS only
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public TlsSettings xtlsSettings { get; set; }
|
public TlsSettings realitySettings { get; set; }
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// grpc
|
/// grpc
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -404,24 +404,23 @@ namespace v2rayN.Mode
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// 是否允许不安全连接(用于客户端)
|
/// 是否允许不安全连接(用于客户端)
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public bool allowInsecure { get; set; }
|
public bool? allowInsecure { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string serverName { get; set; }
|
public string? serverName { get; set; }
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public List<string> alpn
|
public List<string>? alpn { get; set; }
|
||||||
{
|
|
||||||
get; set;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
public string? fingerprint { get; set; }
|
||||||
/// "chrome" | "firefox" | "safari" | "randomized"
|
|
||||||
/// </summary>
|
public bool? show { get; set; } = false;
|
||||||
public string fingerprint { get; set; }
|
public string? publicKey { get; set; }
|
||||||
|
public string? shortId { get; set; }
|
||||||
|
public string? spiderX { get; set; }
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
83
v2rayN/v2rayN/Resx/ResUI.Designer.cs
generated
83
v2rayN/v2rayN/Resx/ResUI.Designer.cs
generated
@@ -816,6 +816,15 @@ namespace v2rayN.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Move up and down 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string menuMoveTo {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("menuMoveTo", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Move to group 的本地化字符串。
|
/// 查找类似 Move to group 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -1375,7 +1384,7 @@ namespace v2rayN.Resx {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Please fill in the address (Url) 的本地化字符串。
|
/// 查找类似 Please fill in the Url 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static string MsgNeedUrl {
|
public static string MsgNeedUrl {
|
||||||
get {
|
get {
|
||||||
@@ -1437,6 +1446,15 @@ namespace v2rayN.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Updates are not enabled, skip this subscription 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string MsgSkipSubscriptionUpdate {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("MsgSkipSubscriptionUpdate", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Start getting subscriptions 的本地化字符串。
|
/// 查找类似 Start getting subscriptions 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -1779,6 +1797,15 @@ namespace v2rayN.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Waiting for testing 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string SpeedtestingWait {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("SpeedtestingWait", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 PAC failed to start. Please run this program as Administrator. 的本地化字符串。
|
/// 查找类似 PAC failed to start. Please run this program as Administrator. 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -2158,6 +2185,15 @@ namespace v2rayN.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 PublicKey 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string TbPublicKey {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("TbPublicKey", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Alias (remarks) 的本地化字符串。
|
/// 查找类似 Alias (remarks) 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -2491,6 +2527,15 @@ namespace v2rayN.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Enable hardware acceleration(Require restart) 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string TbSettingsEnableHWA {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("TbSettingsEnableHWA", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Exception 的本地化字符串。
|
/// 查找类似 Exception 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -2833,6 +2878,15 @@ namespace v2rayN.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 DNS object, e.g. {"servers":[]} 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string TbSettingsTunModeDNS {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("TbSettingsTunModeDNS", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 Proxy IP CIDR, separated by commas (,) 的本地化字符串。
|
/// 查找类似 Proxy IP CIDR, separated by commas (,) 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -2878,6 +2932,15 @@ namespace v2rayN.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 ShortId 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string TbShortId {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("TbShortId", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 SNI 的本地化字符串。
|
/// 查找类似 SNI 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -2887,6 +2950,15 @@ namespace v2rayN.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 SpiderX 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string TbSpiderX {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("TbSpiderX", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 TLS 的本地化字符串。
|
/// 查找类似 TLS 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@@ -2932,6 +3004,15 @@ namespace v2rayN.Resx {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 查找类似 Please turn off when there is an abnormal disconnection 的本地化字符串。
|
||||||
|
/// </summary>
|
||||||
|
public static string TipDisplayLog {
|
||||||
|
get {
|
||||||
|
return ResourceManager.GetString("TipDisplayLog", resourceCulture);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 查找类似 *Default value tcp 的本地化字符串。
|
/// 查找类似 *Default value tcp 的本地化字符串。
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
|||||||
@@ -1066,4 +1066,7 @@
|
|||||||
<data name="TbSettingsDefUserAgentTips" xml:space="preserve">
|
<data name="TbSettingsDefUserAgentTips" xml:space="preserve">
|
||||||
<value>This parameter is valid only for tcp/http and ws</value>
|
<value>This parameter is valid only for tcp/http and ws</value>
|
||||||
</data>
|
</data>
|
||||||
|
<data name="TbSettingsEnableHWA" xml:space="preserve">
|
||||||
|
<value>فعالسازی شتابدهنده سختافزاری (نیاز به راهاندازی مجدد)</value>
|
||||||
|
</data>
|
||||||
</root>
|
</root>
|
||||||
@@ -380,7 +380,7 @@
|
|||||||
<value>Count</value>
|
<value>Count</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MsgNeedUrl" xml:space="preserve">
|
<data name="MsgNeedUrl" xml:space="preserve">
|
||||||
<value>Please fill in the address (Url)</value>
|
<value>Please fill in the Url</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="AddBatchRoutingRulesYesNo" xml:space="preserve">
|
<data name="AddBatchRoutingRulesYesNo" xml:space="preserve">
|
||||||
<value>Do you want to append rules? Choose yes to append, choose otherwise to replace</value>
|
<value>Do you want to append rules? Choose yes to append, choose otherwise to replace</value>
|
||||||
@@ -1126,4 +1126,31 @@
|
|||||||
<data name="TbSettingsSpeedTestUrl" xml:space="preserve">
|
<data name="TbSettingsSpeedTestUrl" xml:space="preserve">
|
||||||
<value>SpeedTest Url</value>
|
<value>SpeedTest Url</value>
|
||||||
</data>
|
</data>
|
||||||
|
<data name="TbSettingsTunModeDNS" xml:space="preserve">
|
||||||
|
<value>DNS object, e.g. {"servers":[]}</value>
|
||||||
|
</data>
|
||||||
|
<data name="menuMoveTo" xml:space="preserve">
|
||||||
|
<value>Move up and down</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbPublicKey" xml:space="preserve">
|
||||||
|
<value>PublicKey</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbShortId" xml:space="preserve">
|
||||||
|
<value>ShortId</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSpiderX" xml:space="preserve">
|
||||||
|
<value>SpiderX</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsEnableHWA" xml:space="preserve">
|
||||||
|
<value>Enable hardware acceleration(Require restart)</value>
|
||||||
|
</data>
|
||||||
|
<data name="SpeedtestingWait" xml:space="preserve">
|
||||||
|
<value>Waiting for testing</value>
|
||||||
|
</data>
|
||||||
|
<data name="TipDisplayLog" xml:space="preserve">
|
||||||
|
<value>Please turn off when there is an abnormal disconnection</value>
|
||||||
|
</data>
|
||||||
|
<data name="MsgSkipSubscriptionUpdate" xml:space="preserve">
|
||||||
|
<value>Updates are not enabled, skip this subscription</value>
|
||||||
|
</data>
|
||||||
</root>
|
</root>
|
||||||
@@ -1108,4 +1108,7 @@
|
|||||||
<data name="TbSettingsFontSize" xml:space="preserve">
|
<data name="TbSettingsFontSize" xml:space="preserve">
|
||||||
<value>Размер шрифта</value>
|
<value>Размер шрифта</value>
|
||||||
</data>
|
</data>
|
||||||
|
<data name="TbSettingsEnableHWA" xml:space="preserve">
|
||||||
|
<value>Включить аппаратное ускорение (требуется перезагрузка)</value>
|
||||||
|
</data>
|
||||||
</root>
|
</root>
|
||||||
@@ -304,7 +304,7 @@
|
|||||||
<value>操作失败,请检查重试</value>
|
<value>操作失败,请检查重试</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="PleaseFillRemarks" xml:space="preserve">
|
<data name="PleaseFillRemarks" xml:space="preserve">
|
||||||
<value>请填写备注</value>
|
<value>请填写别名</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="PleaseSelectEncryption" xml:space="preserve">
|
<data name="PleaseSelectEncryption" xml:space="preserve">
|
||||||
<value>请选择加密方式</value>
|
<value>请选择加密方式</value>
|
||||||
@@ -380,7 +380,7 @@
|
|||||||
<value>数量</value>
|
<value>数量</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="MsgNeedUrl" xml:space="preserve">
|
<data name="MsgNeedUrl" xml:space="preserve">
|
||||||
<value>请填写地址(Url)</value>
|
<value>请填写Url</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="AddBatchRoutingRulesYesNo" xml:space="preserve">
|
<data name="AddBatchRoutingRulesYesNo" xml:space="preserve">
|
||||||
<value>是否追加规则?选择是则追加,选择否则替换</value>
|
<value>是否追加规则?选择是则追加,选择否则替换</value>
|
||||||
@@ -710,7 +710,7 @@
|
|||||||
<value>SNI</value>
|
<value>SNI</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TbStreamSecurity" xml:space="preserve">
|
<data name="TbStreamSecurity" xml:space="preserve">
|
||||||
<value>TLS</value>
|
<value>传输层安全(TLS)</value>
|
||||||
</data>
|
</data>
|
||||||
<data name="TipNetwork" xml:space="preserve">
|
<data name="TipNetwork" xml:space="preserve">
|
||||||
<value>*默认tcp,选错会无法连接</value>
|
<value>*默认tcp,选错会无法连接</value>
|
||||||
@@ -1126,4 +1126,31 @@
|
|||||||
<data name="TbSettingsSpeedTestUrl" xml:space="preserve">
|
<data name="TbSettingsSpeedTestUrl" xml:space="preserve">
|
||||||
<value>测速文件地址</value>
|
<value>测速文件地址</value>
|
||||||
</data>
|
</data>
|
||||||
</root>
|
<data name="TbSettingsTunModeDNS" xml:space="preserve">
|
||||||
|
<value>DNS对象,例如 {"servers":[]}</value>
|
||||||
|
</data>
|
||||||
|
<data name="menuMoveTo" xml:space="preserve">
|
||||||
|
<value>移至上下</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbPublicKey" xml:space="preserve">
|
||||||
|
<value>PublicKey</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbShortId" xml:space="preserve">
|
||||||
|
<value>ShortId</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSpiderX" xml:space="preserve">
|
||||||
|
<value>SpiderX</value>
|
||||||
|
</data>
|
||||||
|
<data name="TbSettingsEnableHWA" xml:space="preserve">
|
||||||
|
<value>启用硬件加速(需重启)</value>
|
||||||
|
</data>
|
||||||
|
<data name="SpeedtestingWait" xml:space="preserve">
|
||||||
|
<value>等待测试中...</value>
|
||||||
|
</data>
|
||||||
|
<data name="TipDisplayLog" xml:space="preserve">
|
||||||
|
<value>当有异常断流时请关闭</value>
|
||||||
|
</data>
|
||||||
|
<data name="MsgSkipSubscriptionUpdate" xml:space="preserve">
|
||||||
|
<value>未启用更新,跳过此订阅</value>
|
||||||
|
</data>
|
||||||
|
</root>
|
||||||
@@ -14,7 +14,14 @@
|
|||||||
{
|
{
|
||||||
"outboundTag": "proxy",
|
"outboundTag": "proxy",
|
||||||
"ip": [
|
"ip": [
|
||||||
"geoip:telegram"
|
"geoip:cloudflare",
|
||||||
|
"geoip:cloudfront",
|
||||||
|
"geoip:facebook",
|
||||||
|
"geoip:fastly",
|
||||||
|
"geoip:google",
|
||||||
|
"geoip:netflix",
|
||||||
|
"geoip:telegram",
|
||||||
|
"geoip:twitter"
|
||||||
],
|
],
|
||||||
"domain": [
|
"domain": [
|
||||||
"geosite:gfw",
|
"geosite:gfw",
|
||||||
|
|||||||
@@ -5,27 +5,7 @@
|
|||||||
$log_output$
|
$log_output$
|
||||||
"timestamp": true
|
"timestamp": true
|
||||||
},
|
},
|
||||||
"dns": {
|
"dns": $dns_object$ ,
|
||||||
"servers": [
|
|
||||||
{
|
|
||||||
"tag": "local",
|
|
||||||
"address": "223.5.5.5",
|
|
||||||
"detour": "direct"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"tag": "block",
|
|
||||||
"address": "rcode://success"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"rules": [
|
|
||||||
{
|
|
||||||
"geosite": "category-ads-all",
|
|
||||||
"server": "block",
|
|
||||||
"disable_cache": true
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"strategy": "ipv4_only"
|
|
||||||
},
|
|
||||||
"inbounds": [
|
"inbounds": [
|
||||||
{
|
{
|
||||||
"type": "tun",
|
"type": "tun",
|
||||||
|
|||||||
31
v2rayN/v2rayN/Sample/tun_singbox_dns
Normal file
31
v2rayN/v2rayN/Sample/tun_singbox_dns
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
{
|
||||||
|
"servers": [
|
||||||
|
{
|
||||||
|
"tag": "out_dns",
|
||||||
|
"address": "8.8.8.8",
|
||||||
|
"detour": "proxy"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"tag": "local",
|
||||||
|
"address": "223.5.5.5",
|
||||||
|
"detour": "direct"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"tag": "block",
|
||||||
|
"address": "rcode://success"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"rules": [
|
||||||
|
{
|
||||||
|
"geosite": "cn",
|
||||||
|
"server": "local",
|
||||||
|
"disable_cache": true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"geosite": "category-ads-all",
|
||||||
|
"server": "block",
|
||||||
|
"disable_cache": true
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"strategy": "ipv4_only"
|
||||||
|
}
|
||||||
@@ -17,6 +17,13 @@ namespace v2rayN.Tool
|
|||||||
config.LoggingRules.Add(new LoggingRule("*", LogLevel.Debug, fileTarget));
|
config.LoggingRules.Add(new LoggingRule("*", LogLevel.Debug, fileTarget));
|
||||||
LogManager.Configuration = config;
|
LogManager.Configuration = config;
|
||||||
}
|
}
|
||||||
|
public static void LoggingEnabled(bool enable)
|
||||||
|
{
|
||||||
|
if (!enable)
|
||||||
|
{
|
||||||
|
LogManager.SuspendLogging();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public static void ClearLogs()
|
public static void ClearLogs()
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -65,24 +65,18 @@ namespace v2rayN
|
|||||||
/// 取得存储资源
|
/// 取得存储资源
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static string LoadResource(string res)
|
public static string? LoadResource(string res)
|
||||||
{
|
{
|
||||||
string result = string.Empty;
|
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (!File.Exists(res))
|
if (!File.Exists(res)) return null;
|
||||||
{
|
return File.ReadAllText(res);
|
||||||
return result;
|
|
||||||
}
|
|
||||||
using StreamReader reader = new(res);
|
|
||||||
result = reader.ReadToEnd();
|
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
SaveLog(ex.Message, ex);
|
SaveLog(ex.Message, ex);
|
||||||
}
|
}
|
||||||
return result;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@@ -91,7 +85,7 @@ namespace v2rayN
|
|||||||
/// <typeparam name="T"></typeparam>
|
/// <typeparam name="T"></typeparam>
|
||||||
/// <param name="strJson"></param>
|
/// <param name="strJson"></param>
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static T? FromJson<T>(string strJson)
|
public static T? FromJson<T>(string? strJson)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@@ -117,6 +111,10 @@ namespace v2rayN
|
|||||||
string result = string.Empty;
|
string result = string.Empty;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
if (obj == null)
|
||||||
|
{
|
||||||
|
return result;
|
||||||
|
}
|
||||||
if (indented)
|
if (indented)
|
||||||
{
|
{
|
||||||
result = JsonConvert.SerializeObject(obj,
|
result = JsonConvert.SerializeObject(obj,
|
||||||
@@ -172,8 +170,7 @@ namespace v2rayN
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
JObject obj = JObject.Parse(strJson);
|
return JObject.Parse(strJson);
|
||||||
return obj;
|
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
@@ -204,7 +201,7 @@ namespace v2rayN
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
return string.Join(",", lst.ToArray());
|
return string.Join(",", lst);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
@@ -285,6 +282,8 @@ namespace v2rayN
|
|||||||
.Replace(Environment.NewLine, "")
|
.Replace(Environment.NewLine, "")
|
||||||
.Replace("\n", "")
|
.Replace("\n", "")
|
||||||
.Replace("\r", "")
|
.Replace("\r", "")
|
||||||
|
.Replace('_', '/')
|
||||||
|
.Replace('-', '+')
|
||||||
.Replace(" ", "");
|
.Replace(" ", "");
|
||||||
|
|
||||||
if (plainText.Length % 4 > 0)
|
if (plainText.Length % 4 > 0)
|
||||||
@@ -315,7 +314,7 @@ namespace v2rayN
|
|||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
SaveLog(ex.Message, ex);
|
//SaveLog(ex.Message, ex);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -327,7 +326,7 @@ namespace v2rayN
|
|||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
SaveLog(ex.Message, ex);
|
//SaveLog(ex.Message, ex);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -340,7 +339,7 @@ namespace v2rayN
|
|||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
SaveLog(ex.Message, ex);
|
//SaveLog(ex.Message, ex);
|
||||||
return string.Empty;
|
return string.Empty;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -489,7 +488,7 @@ namespace v2rayN
|
|||||||
{
|
{
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (text.Equals("null"))
|
if (text == "null")
|
||||||
{
|
{
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@@ -636,7 +635,7 @@ namespace v2rayN
|
|||||||
|
|
||||||
string value = RegReadValue(Global.AutoRunRegPath, Global.AutoRunName, "");
|
string value = RegReadValue(Global.AutoRunRegPath, Global.AutoRunName, "");
|
||||||
string exePath = GetExePath();
|
string exePath = GetExePath();
|
||||||
if (value?.Equals(exePath) == true || value?.Equals($"\"{exePath}\"") == true)
|
if (value == exePath || value == $"\"{exePath}\"")
|
||||||
{
|
{
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@@ -738,9 +737,9 @@ namespace v2rayN
|
|||||||
{
|
{
|
||||||
const string subkey = @"SOFTWARE\Microsoft\NET Framework Setup\NDP\v4\Full\";
|
const string subkey = @"SOFTWARE\Microsoft\NET Framework Setup\NDP\v4\Full\";
|
||||||
using RegistryKey? ndpKey = RegistryKey.OpenBaseKey(RegistryHive.LocalMachine, RegistryView.Registry32).OpenSubKey(subkey);
|
using RegistryKey? ndpKey = RegistryKey.OpenBaseKey(RegistryHive.LocalMachine, RegistryView.Registry32).OpenSubKey(subkey);
|
||||||
if (ndpKey != null && ndpKey.GetValue("Release") != null)
|
if (ndpKey?.GetValue("Release") != null)
|
||||||
{
|
{
|
||||||
return (int)ndpKey.GetValue("Release") >= release ? true : false;
|
return (int)ndpKey.GetValue("Release") >= release;
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -781,7 +780,7 @@ namespace v2rayN
|
|||||||
task.Settings.RunOnlyIfIdle = false;
|
task.Settings.RunOnlyIfIdle = false;
|
||||||
task.Settings.IdleSettings.StopOnIdleEnd = false;
|
task.Settings.IdleSettings.StopOnIdleEnd = false;
|
||||||
task.Settings.ExecutionTimeLimit = TimeSpan.Zero;
|
task.Settings.ExecutionTimeLimit = TimeSpan.Zero;
|
||||||
task.Triggers.Add(new LogonTrigger { UserId = logonUser, Delay = TimeSpan.FromMinutes(1) });
|
task.Triggers.Add(new LogonTrigger { UserId = logonUser, Delay = TimeSpan.FromSeconds(10) });
|
||||||
task.Principal.RunLevel = TaskRunLevel.Highest;
|
task.Principal.RunLevel = TaskRunLevel.Highest;
|
||||||
task.Actions.Add(new ExecAction(deamonFileName));
|
task.Actions.Add(new ExecAction(deamonFileName));
|
||||||
|
|
||||||
@@ -910,13 +909,13 @@ namespace v2rayN
|
|||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static string? GetClipboardData()
|
public static string? GetClipboardData()
|
||||||
{
|
{
|
||||||
string strData = string.Empty;
|
string? strData = string.Empty;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
IDataObject data = Clipboard.GetDataObject();
|
IDataObject data = Clipboard.GetDataObject();
|
||||||
if (data.GetDataPresent(DataFormats.UnicodeText))
|
if (data.GetDataPresent(DataFormats.UnicodeText))
|
||||||
{
|
{
|
||||||
strData = data.GetData(DataFormats.UnicodeText).ToString();
|
strData = data.GetData(DataFormats.UnicodeText)?.ToString();
|
||||||
}
|
}
|
||||||
return strData;
|
return strData;
|
||||||
}
|
}
|
||||||
@@ -1153,17 +1152,23 @@ namespace v2rayN
|
|||||||
|
|
||||||
public static void SaveLog(string strContent)
|
public static void SaveLog(string strContent)
|
||||||
{
|
{
|
||||||
var logger = LogManager.GetLogger("Log1");
|
if (LogManager.IsLoggingEnabled())
|
||||||
logger.Info(strContent);
|
{
|
||||||
|
var logger = LogManager.GetLogger("Log1");
|
||||||
|
logger.Info(strContent);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
public static void SaveLog(string strTitle, Exception ex)
|
public static void SaveLog(string strTitle, Exception ex)
|
||||||
{
|
{
|
||||||
var logger = LogManager.GetLogger("Log2");
|
if (LogManager.IsLoggingEnabled())
|
||||||
logger.Debug($"{strTitle},{ex.Message}");
|
|
||||||
logger.Debug(ex.StackTrace);
|
|
||||||
if (ex != null && ex.InnerException != null)
|
|
||||||
{
|
{
|
||||||
logger.Error(ex.InnerException);
|
var logger = LogManager.GetLogger("Log2");
|
||||||
|
logger.Debug($"{strTitle},{ex.Message}");
|
||||||
|
logger.Debug(ex.StackTrace);
|
||||||
|
if (ex?.InnerException != null)
|
||||||
|
{
|
||||||
|
logger.Error(ex.InnerException);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1193,8 +1198,8 @@ namespace v2rayN
|
|||||||
{
|
{
|
||||||
int marginLeft = (int)((double)fullImage.Width * i / 2.5 / maxTry);
|
int marginLeft = (int)((double)fullImage.Width * i / 2.5 / maxTry);
|
||||||
int marginTop = (int)((double)fullImage.Height * i / 2.5 / maxTry);
|
int marginTop = (int)((double)fullImage.Height * i / 2.5 / maxTry);
|
||||||
Rectangle cropRect = new Rectangle(marginLeft, marginTop, fullImage.Width - marginLeft * 2, fullImage.Height - marginTop * 2);
|
Rectangle cropRect = new(marginLeft, marginTop, fullImage.Width - marginLeft * 2, fullImage.Height - marginTop * 2);
|
||||||
Bitmap target = new Bitmap(screen.Bounds.Width, screen.Bounds.Height);
|
Bitmap target = new(screen.Bounds.Width, screen.Bounds.Height);
|
||||||
|
|
||||||
double imageScale = (double)screen.Bounds.Width / (double)cropRect.Width;
|
double imageScale = (double)screen.Bounds.Width / (double)cropRect.Width;
|
||||||
using (Graphics g = Graphics.FromImage(target))
|
using (Graphics g = Graphics.FromImage(target))
|
||||||
@@ -1204,9 +1209,9 @@ namespace v2rayN
|
|||||||
GraphicsUnit.Pixel);
|
GraphicsUnit.Pixel);
|
||||||
}
|
}
|
||||||
|
|
||||||
BitmapLuminanceSource source = new BitmapLuminanceSource(target);
|
BitmapLuminanceSource source = new(target);
|
||||||
BinaryBitmap bitmap = new BinaryBitmap(new HybridBinarizer(source));
|
BinaryBitmap bitmap = new(new HybridBinarizer(source));
|
||||||
QRCodeReader reader = new QRCodeReader();
|
QRCodeReader reader = new();
|
||||||
Result result = reader.decode(bitmap);
|
Result result = reader.decode(bitmap);
|
||||||
if (result != null)
|
if (result != null)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -118,6 +118,10 @@ namespace v2rayN.ViewModels
|
|||||||
item.allowInsecure = SelectedSource.allowInsecure;
|
item.allowInsecure = SelectedSource.allowInsecure;
|
||||||
item.fingerprint = SelectedSource.fingerprint;
|
item.fingerprint = SelectedSource.fingerprint;
|
||||||
item.alpn = SelectedSource.alpn;
|
item.alpn = SelectedSource.alpn;
|
||||||
|
|
||||||
|
item.publicKey = SelectedSource.publicKey;
|
||||||
|
item.shortId = SelectedSource.shortId;
|
||||||
|
item.spiderX = SelectedSource.spiderX;
|
||||||
}
|
}
|
||||||
|
|
||||||
int ret = -1;
|
int ret = -1;
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ using DynamicData.Binding;
|
|||||||
using MaterialDesignColors;
|
using MaterialDesignColors;
|
||||||
using MaterialDesignColors.ColorManipulation;
|
using MaterialDesignColors.ColorManipulation;
|
||||||
using MaterialDesignThemes.Wpf;
|
using MaterialDesignThemes.Wpf;
|
||||||
using NHotkey;
|
|
||||||
using ReactiveUI;
|
using ReactiveUI;
|
||||||
using ReactiveUI.Fody.Helpers;
|
using ReactiveUI.Fody.Helpers;
|
||||||
using Splat;
|
using Splat;
|
||||||
@@ -38,7 +37,7 @@ namespace v2rayN.ViewModels
|
|||||||
private static Config _config;
|
private static Config _config;
|
||||||
private NoticeHandler? _noticeHandler;
|
private NoticeHandler? _noticeHandler;
|
||||||
private readonly PaletteHelper _paletteHelper = new();
|
private readonly PaletteHelper _paletteHelper = new();
|
||||||
private Dictionary<int, bool> _dicHeaderSort = new();
|
private Dictionary<string, bool> _dicHeaderSort = new();
|
||||||
private Action<string> _updateView;
|
private Action<string> _updateView;
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
@@ -70,6 +69,8 @@ namespace v2rayN.ViewModels
|
|||||||
public ComboItem SelectedServer { get; set; }
|
public ComboItem SelectedServer { get; set; }
|
||||||
[Reactive]
|
[Reactive]
|
||||||
public string ServerFilter { get; set; }
|
public string ServerFilter { get; set; }
|
||||||
|
[Reactive]
|
||||||
|
public bool BlServers { get; set; }
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Menu
|
#region Menu
|
||||||
@@ -175,6 +176,8 @@ namespace v2rayN.ViewModels
|
|||||||
[Reactive]
|
[Reactive]
|
||||||
public string RunningServerDisplay { get; set; }
|
public string RunningServerDisplay { get; set; }
|
||||||
[Reactive]
|
[Reactive]
|
||||||
|
public string RunningServerToolTipText { get; set; }
|
||||||
|
[Reactive]
|
||||||
public string RunningInfoDisplay { get; set; }
|
public string RunningInfoDisplay { get; set; }
|
||||||
[Reactive]
|
[Reactive]
|
||||||
public string SpeedProxyDisplay { get; set; }
|
public string SpeedProxyDisplay { get; set; }
|
||||||
@@ -218,10 +221,11 @@ namespace v2rayN.ViewModels
|
|||||||
{
|
{
|
||||||
EnableTun = true;
|
EnableTun = true;
|
||||||
}
|
}
|
||||||
|
_subId = _config.subIndexId;
|
||||||
|
|
||||||
//RefreshServers();
|
|
||||||
InitSubscriptionView();
|
InitSubscriptionView();
|
||||||
RefreshRoutingsMenu();
|
RefreshRoutingsMenu();
|
||||||
|
RefreshServers();
|
||||||
|
|
||||||
var canEditRemove = this.WhenAnyValue(
|
var canEditRemove = this.WhenAnyValue(
|
||||||
x => x.SelectedProfile,
|
x => x.SelectedProfile,
|
||||||
@@ -365,7 +369,7 @@ namespace v2rayN.ViewModels
|
|||||||
}, canEditRemove);
|
}, canEditRemove);
|
||||||
SortServerResultCmd = ReactiveCommand.Create(() =>
|
SortServerResultCmd = ReactiveCommand.Create(() =>
|
||||||
{
|
{
|
||||||
SortServer((int)EServerColName.delay);
|
SortServer(EServerColName.delayVal.ToString());
|
||||||
});
|
});
|
||||||
//servers export
|
//servers export
|
||||||
Export2ClientConfigCmd = ReactiveCommand.Create(() =>
|
Export2ClientConfigCmd = ReactiveCommand.Create(() =>
|
||||||
@@ -537,8 +541,12 @@ namespace v2rayN.ViewModels
|
|||||||
_noticeHandler?.SendMessage(msg);
|
_noticeHandler?.SendMessage(msg);
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
|
var indexIdOld = _config.indexId;
|
||||||
RefreshServers();
|
RefreshServers();
|
||||||
Reload();
|
if (indexIdOld != _config.indexId)
|
||||||
|
{
|
||||||
|
Reload();
|
||||||
|
}
|
||||||
if (_config.uiItem.enableAutoAdjustMainLvColWidth)
|
if (_config.uiItem.enableAutoAdjustMainLvColWidth)
|
||||||
{
|
{
|
||||||
_updateView("AdjustMainLvColWidth");
|
_updateView("AdjustMainLvColWidth");
|
||||||
@@ -623,27 +631,26 @@ namespace v2rayN.ViewModels
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnHotkeyHandler(object sender, HotkeyEventArgs e)
|
private void OnHotkeyHandler(EGlobalHotkey e)
|
||||||
{
|
{
|
||||||
switch (Utils.ToInt(e.Name))
|
switch (e)
|
||||||
{
|
{
|
||||||
case (int)EGlobalHotkey.ShowForm:
|
case EGlobalHotkey.ShowForm:
|
||||||
ShowHideWindow(null);
|
ShowHideWindow(null);
|
||||||
break;
|
break;
|
||||||
case (int)EGlobalHotkey.SystemProxyClear:
|
case EGlobalHotkey.SystemProxyClear:
|
||||||
SetListenerType(ESysProxyType.ForcedClear);
|
SetListenerType(ESysProxyType.ForcedClear);
|
||||||
break;
|
break;
|
||||||
case (int)EGlobalHotkey.SystemProxySet:
|
case EGlobalHotkey.SystemProxySet:
|
||||||
SetListenerType(ESysProxyType.ForcedChange);
|
SetListenerType(ESysProxyType.ForcedChange);
|
||||||
break;
|
break;
|
||||||
case (int)EGlobalHotkey.SystemProxyUnchanged:
|
case EGlobalHotkey.SystemProxyUnchanged:
|
||||||
SetListenerType(ESysProxyType.Unchanged);
|
SetListenerType(ESysProxyType.Unchanged);
|
||||||
break;
|
break;
|
||||||
case (int)EGlobalHotkey.SystemProxyPac:
|
case EGlobalHotkey.SystemProxyPac:
|
||||||
SetListenerType(ESysProxyType.Pac);
|
SetListenerType(ESysProxyType.Pac);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
e.Handled = true;
|
|
||||||
}
|
}
|
||||||
public void MyAppExit(bool blWindowsShutDown)
|
public void MyAppExit(bool blWindowsShutDown)
|
||||||
{
|
{
|
||||||
@@ -691,6 +698,7 @@ namespace v2rayN.ViewModels
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
_subId = SelectedSub?.id;
|
_subId = SelectedSub?.id;
|
||||||
|
_config.subIndexId = _subId;
|
||||||
|
|
||||||
RefreshServers();
|
RefreshServers();
|
||||||
|
|
||||||
@@ -710,6 +718,7 @@ namespace v2rayN.ViewModels
|
|||||||
private void RefreshServers()
|
private void RefreshServers()
|
||||||
{
|
{
|
||||||
List<ProfileItemModel> lstModel = LazyConfig.Instance.ProfileItems(_subId, _serverFilter);
|
List<ProfileItemModel> lstModel = LazyConfig.Instance.ProfileItems(_subId, _serverFilter);
|
||||||
|
|
||||||
ConfigHandler.SetDefaultServer(_config, lstModel);
|
ConfigHandler.SetDefaultServer(_config, lstModel);
|
||||||
|
|
||||||
List<ServerStatItem> lstServerStat = new();
|
List<ServerStatItem> lstServerStat = new();
|
||||||
@@ -733,9 +742,11 @@ namespace v2rayN.ViewModels
|
|||||||
security = t.security,
|
security = t.security,
|
||||||
network = t.network,
|
network = t.network,
|
||||||
streamSecurity = t.streamSecurity,
|
streamSecurity = t.streamSecurity,
|
||||||
|
subid = t.subid,
|
||||||
subRemarks = t.subRemarks,
|
subRemarks = t.subRemarks,
|
||||||
isActive = t.indexId == _config.indexId,
|
isActive = t.indexId == _config.indexId,
|
||||||
sort = t33 == null ? 0 : t33.sort,
|
sort = t33 == null ? 0 : t33.sort,
|
||||||
|
delay = t33 == null ? 0 : t33.delay,
|
||||||
delayVal = t33?.delay != 0 ? $"{t33?.delay} {Global.DelayUnit}" : string.Empty,
|
delayVal = t33?.delay != 0 ? $"{t33?.delay} {Global.DelayUnit}" : string.Empty,
|
||||||
speedVal = t33?.speed != 0 ? $"{t33?.speed} {Global.SpeedUnit}" : string.Empty,
|
speedVal = t33?.speed != 0 ? $"{t33?.speed} {Global.SpeedUnit}" : string.Empty,
|
||||||
todayDown = t22 == null ? "" : Utils.HumanFy(t22.todayDown),
|
todayDown = t22 == null ? "" : Utils.HumanFy(t22.todayDown),
|
||||||
@@ -768,7 +779,14 @@ namespace v2rayN.ViewModels
|
|||||||
var running = ConfigHandler.GetDefaultServer(ref _config);
|
var running = ConfigHandler.GetDefaultServer(ref _config);
|
||||||
if (running != null)
|
if (running != null)
|
||||||
{
|
{
|
||||||
RunningServerDisplay = string.Format("{0}:{1}", ResUI.menuServers, running.GetSummary());
|
var runningSummary = running.GetSummary();
|
||||||
|
RunningServerDisplay = $"{ResUI.menuServers}:{runningSummary}";
|
||||||
|
RunningServerToolTipText = runningSummary;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
RunningServerDisplay =
|
||||||
|
RunningServerToolTipText = ResUI.CheckServerSettings;
|
||||||
}
|
}
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
@@ -778,9 +796,11 @@ namespace v2rayN.ViewModels
|
|||||||
_servers.Clear();
|
_servers.Clear();
|
||||||
if (_lstProfile.Count > _config.guiItem.trayMenuServersLimit)
|
if (_lstProfile.Count > _config.guiItem.trayMenuServersLimit)
|
||||||
{
|
{
|
||||||
|
BlServers = false;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BlServers = true;
|
||||||
for (int k = 0; k < _lstProfile.Count; k++)
|
for (int k = 0; k < _lstProfile.Count; k++)
|
||||||
{
|
{
|
||||||
ProfileItem it = _lstProfile[k];
|
ProfileItem it = _lstProfile[k];
|
||||||
@@ -788,7 +808,7 @@ namespace v2rayN.ViewModels
|
|||||||
|
|
||||||
var item = new ComboItem() { ID = it.indexId, Text = name };
|
var item = new ComboItem() { ID = it.indexId, Text = name };
|
||||||
_servers.Add(item);
|
_servers.Add(item);
|
||||||
if (_config.indexId.Equals(it.indexId))
|
if (_config.indexId == it.indexId)
|
||||||
{
|
{
|
||||||
SelectedServer = item;
|
SelectedServer = item;
|
||||||
}
|
}
|
||||||
@@ -817,21 +837,31 @@ namespace v2rayN.ViewModels
|
|||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Add Servers
|
#region Add Servers
|
||||||
private int GetProfileItems(out List<ProfileItem> lstSelecteds)
|
private int GetProfileItems(out List<ProfileItem> lstSelecteds, bool latest)
|
||||||
{
|
{
|
||||||
lstSelecteds = new List<ProfileItem>();
|
lstSelecteds = new List<ProfileItem>();
|
||||||
if (SelectedProfiles == null || SelectedProfiles.Count <= 0)
|
if (SelectedProfiles == null || SelectedProfiles.Count <= 0)
|
||||||
{
|
{
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
foreach (var profile in SelectedProfiles)
|
|
||||||
|
var orderProfiles = SelectedProfiles?.OrderBy(t => t.sort);
|
||||||
|
if (latest)
|
||||||
{
|
{
|
||||||
var item = LazyConfig.Instance.GetProfileItem(profile.indexId);
|
foreach (var profile in orderProfiles)
|
||||||
if (item is not null)
|
|
||||||
{
|
{
|
||||||
lstSelecteds.Add(item);
|
var item = LazyConfig.Instance.GetProfileItem(profile.indexId);
|
||||||
|
if (item is not null)
|
||||||
|
{
|
||||||
|
lstSelecteds.Add(item);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
lstSelecteds = Utils.FromJson<List<ProfileItem>>(Utils.ToJson(orderProfiles));
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -918,7 +948,7 @@ namespace v2rayN.ViewModels
|
|||||||
}
|
}
|
||||||
public void RemoveServer()
|
public void RemoveServer()
|
||||||
{
|
{
|
||||||
if (GetProfileItems(out List<ProfileItem> lstSelecteds) < 0)
|
if (GetProfileItems(out List<ProfileItem> lstSelecteds, false) < 0)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -941,15 +971,14 @@ namespace v2rayN.ViewModels
|
|||||||
|
|
||||||
private void RemoveDuplicateServer()
|
private void RemoveDuplicateServer()
|
||||||
{
|
{
|
||||||
int oldCount = _lstProfile.Count;
|
var tuple = ConfigHandler.DedupServerList(_config, _subId);
|
||||||
int newCount = ConfigHandler.DedupServerList(ref _config, ref _lstProfile);
|
|
||||||
RefreshServers();
|
RefreshServers();
|
||||||
Reload();
|
Reload();
|
||||||
_noticeHandler?.Enqueue(string.Format(ResUI.RemoveDuplicateServerResult, oldCount, newCount));
|
_noticeHandler?.Enqueue(string.Format(ResUI.RemoveDuplicateServerResult, tuple.Item1, tuple.Item2));
|
||||||
}
|
}
|
||||||
private void CopyServer()
|
private void CopyServer()
|
||||||
{
|
{
|
||||||
if (GetProfileItems(out List<ProfileItem> lstSelecteds) < 0)
|
if (GetProfileItems(out List<ProfileItem> lstSelecteds, false) < 0)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -1032,23 +1061,23 @@ namespace v2rayN.ViewModels
|
|||||||
await DialogHost.Show(dialog, "RootDialog");
|
await DialogHost.Show(dialog, "RootDialog");
|
||||||
}
|
}
|
||||||
|
|
||||||
public void SortServer(int colIndex)
|
public void SortServer(string colName)
|
||||||
{
|
{
|
||||||
if (colIndex < 0)
|
if (Utils.IsNullOrEmpty(colName))
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!_dicHeaderSort.ContainsKey(colIndex))
|
if (!_dicHeaderSort.ContainsKey(colName))
|
||||||
{
|
{
|
||||||
_dicHeaderSort.Add(colIndex, true);
|
_dicHeaderSort.Add(colName, true);
|
||||||
}
|
}
|
||||||
_dicHeaderSort.TryGetValue(colIndex, out bool asc);
|
_dicHeaderSort.TryGetValue(colName, out bool asc);
|
||||||
if (ConfigHandler.SortServers(ref _config, _subId, (EServerColName)colIndex, asc) != 0)
|
if (ConfigHandler.SortServers(ref _config, _subId, colName, asc) != 0)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
_dicHeaderSort[colIndex] = !asc;
|
_dicHeaderSort[colName] = !asc;
|
||||||
RefreshServers();
|
RefreshServers();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1081,7 +1110,7 @@ namespace v2rayN.ViewModels
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (GetProfileItems(out List<ProfileItem> lstSelecteds) < 0)
|
if (GetProfileItems(out List<ProfileItem> lstSelecteds, false) < 0)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -1132,7 +1161,7 @@ namespace v2rayN.ViewModels
|
|||||||
{
|
{
|
||||||
SelectedProfiles = _profileItems;
|
SelectedProfiles = _profileItems;
|
||||||
}
|
}
|
||||||
if (GetProfileItems(out List<ProfileItem> lstSelecteds) < 0)
|
if (GetProfileItems(out List<ProfileItem> lstSelecteds, false) < 0)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -1164,7 +1193,7 @@ namespace v2rayN.ViewModels
|
|||||||
|
|
||||||
public void Export2ShareUrl()
|
public void Export2ShareUrl()
|
||||||
{
|
{
|
||||||
if (GetProfileItems(out List<ProfileItem> lstSelecteds) < 0)
|
if (GetProfileItems(out List<ProfileItem> lstSelecteds, true) < 0)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -1189,7 +1218,7 @@ namespace v2rayN.ViewModels
|
|||||||
|
|
||||||
private void Export2SubContent()
|
private void Export2SubContent()
|
||||||
{
|
{
|
||||||
if (GetProfileItems(out List<ProfileItem> lstSelecteds) < 0)
|
if (GetProfileItems(out List<ProfileItem> lstSelecteds, true) < 0)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -1360,7 +1389,6 @@ namespace v2rayN.ViewModels
|
|||||||
|
|
||||||
public void Reload()
|
public void Reload()
|
||||||
{
|
{
|
||||||
Global.reloadCore = true;
|
|
||||||
_ = LoadV2ray();
|
_ = LoadV2ray();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1380,7 +1408,6 @@ namespace v2rayN.ViewModels
|
|||||||
{
|
{
|
||||||
_coreHandler.LoadCore(_config);
|
_coreHandler.LoadCore(_config);
|
||||||
|
|
||||||
Global.reloadCore = false;
|
|
||||||
//ConfigHandler.SaveConfig(ref _config, false);
|
//ConfigHandler.SaveConfig(ref _config, false);
|
||||||
|
|
||||||
ChangeSystemProxyStatus(_config.sysProxyType, false);
|
ChangeSystemProxyStatus(_config.sysProxyType, false);
|
||||||
@@ -1456,7 +1483,7 @@ namespace v2rayN.ViewModels
|
|||||||
foreach (var item in routings)
|
foreach (var item in routings)
|
||||||
{
|
{
|
||||||
_routingItems.Add(item);
|
_routingItems.Add(item);
|
||||||
if (item.id.Equals(_config.routingBasicItem.routingIndexId))
|
if (item.id == _config.routingBasicItem.routingIndexId)
|
||||||
{
|
{
|
||||||
SelectedRouting = item;
|
SelectedRouting = item;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -65,6 +65,7 @@ namespace v2rayN.ViewModels
|
|||||||
[Reactive] public string currentFontFamily { get; set; }
|
[Reactive] public string currentFontFamily { get; set; }
|
||||||
[Reactive] public int SpeedTestTimeout { get; set; }
|
[Reactive] public int SpeedTestTimeout { get; set; }
|
||||||
[Reactive] public string SpeedTestUrl { get; set; }
|
[Reactive] public string SpeedTestUrl { get; set; }
|
||||||
|
[Reactive] public bool EnableHWA { get; set; }
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
@@ -84,8 +85,10 @@ namespace v2rayN.ViewModels
|
|||||||
[Reactive] public bool TunBypassMode2 { get; set; }
|
[Reactive] public bool TunBypassMode2 { get; set; }
|
||||||
[Reactive] public string TunDirectIP { get; set; }
|
[Reactive] public string TunDirectIP { get; set; }
|
||||||
[Reactive] public string TunDirectProcess { get; set; }
|
[Reactive] public string TunDirectProcess { get; set; }
|
||||||
|
[Reactive] public string TunDirectDNS { get; set; }
|
||||||
[Reactive] public string TunProxyIP { get; set; }
|
[Reactive] public string TunProxyIP { get; set; }
|
||||||
[Reactive] public string TunProxyProcess { get; set; }
|
[Reactive] public string TunProxyProcess { get; set; }
|
||||||
|
[Reactive] public string TunProxyDNS { get; set; }
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region CoreType
|
#region CoreType
|
||||||
@@ -158,6 +161,7 @@ namespace v2rayN.ViewModels
|
|||||||
currentFontFamily = _config.uiItem.currentFontFamily;
|
currentFontFamily = _config.uiItem.currentFontFamily;
|
||||||
SpeedTestTimeout = _config.speedTestItem.speedTestTimeout;
|
SpeedTestTimeout = _config.speedTestItem.speedTestTimeout;
|
||||||
SpeedTestUrl = _config.speedTestItem.speedTestUrl;
|
SpeedTestUrl = _config.speedTestItem.speedTestUrl;
|
||||||
|
EnableHWA = _config.guiItem.enableHWA;
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
@@ -177,8 +181,10 @@ namespace v2rayN.ViewModels
|
|||||||
TunBypassMode = _config.tunModeItem.bypassMode;
|
TunBypassMode = _config.tunModeItem.bypassMode;
|
||||||
TunDirectIP = Utils.List2String(_config.tunModeItem.directIP, true);
|
TunDirectIP = Utils.List2String(_config.tunModeItem.directIP, true);
|
||||||
TunDirectProcess = Utils.List2String(_config.tunModeItem.directProcess, true);
|
TunDirectProcess = Utils.List2String(_config.tunModeItem.directProcess, true);
|
||||||
|
TunDirectDNS = _config.tunModeItem.directDNS;
|
||||||
TunProxyIP = Utils.List2String(_config.tunModeItem.proxyIP, true);
|
TunProxyIP = Utils.List2String(_config.tunModeItem.proxyIP, true);
|
||||||
TunProxyProcess = Utils.List2String(_config.tunModeItem.proxyProcess, true);
|
TunProxyProcess = Utils.List2String(_config.tunModeItem.proxyProcess, true);
|
||||||
|
TunProxyDNS = _config.tunModeItem.proxyDNS;
|
||||||
this.WhenAnyValue(
|
this.WhenAnyValue(
|
||||||
x => x.TunBypassMode)
|
x => x.TunBypassMode)
|
||||||
.Subscribe(c => TunBypassMode2 = !TunBypassMode);
|
.Subscribe(c => TunBypassMode2 = !TunBypassMode);
|
||||||
@@ -253,7 +259,7 @@ namespace v2rayN.ViewModels
|
|||||||
}
|
}
|
||||||
|
|
||||||
var obj = Utils.ParseJson(remoteDNS);
|
var obj = Utils.ParseJson(remoteDNS);
|
||||||
if (obj != null && obj.ContainsKey("servers"))
|
if (obj?.ContainsKey("servers") == true)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -277,7 +283,7 @@ namespace v2rayN.ViewModels
|
|||||||
//}
|
//}
|
||||||
|
|
||||||
//Core
|
//Core
|
||||||
_config.inbound[0].localPort = Utils.ToInt(localPort);
|
_config.inbound[0].localPort = localPort;
|
||||||
_config.inbound[0].udpEnabled = udpEnabled;
|
_config.inbound[0].udpEnabled = udpEnabled;
|
||||||
_config.inbound[0].sniffingEnabled = sniffingEnabled;
|
_config.inbound[0].sniffingEnabled = sniffingEnabled;
|
||||||
_config.inbound[0].routeOnly = routeOnly;
|
_config.inbound[0].routeOnly = routeOnly;
|
||||||
@@ -335,6 +341,7 @@ namespace v2rayN.ViewModels
|
|||||||
_config.uiItem.currentFontFamily = currentFontFamily;
|
_config.uiItem.currentFontFamily = currentFontFamily;
|
||||||
_config.speedTestItem.speedTestTimeout = SpeedTestTimeout;
|
_config.speedTestItem.speedTestTimeout = SpeedTestTimeout;
|
||||||
_config.speedTestItem.speedTestUrl = SpeedTestUrl;
|
_config.speedTestItem.speedTestUrl = SpeedTestUrl;
|
||||||
|
_config.guiItem.enableHWA = EnableHWA;
|
||||||
|
|
||||||
//systemProxy
|
//systemProxy
|
||||||
_config.systemProxyExceptions = systemProxyExceptions;
|
_config.systemProxyExceptions = systemProxyExceptions;
|
||||||
@@ -350,8 +357,10 @@ namespace v2rayN.ViewModels
|
|||||||
_config.tunModeItem.bypassMode = TunBypassMode;
|
_config.tunModeItem.bypassMode = TunBypassMode;
|
||||||
_config.tunModeItem.directIP = Utils.String2List(TunDirectIP);
|
_config.tunModeItem.directIP = Utils.String2List(TunDirectIP);
|
||||||
_config.tunModeItem.directProcess = Utils.String2List(TunDirectProcess);
|
_config.tunModeItem.directProcess = Utils.String2List(TunDirectProcess);
|
||||||
|
_config.tunModeItem.directDNS = Utils.ToJson(Utils.ParseJson(TunDirectDNS));
|
||||||
_config.tunModeItem.proxyIP = Utils.String2List(TunProxyIP);
|
_config.tunModeItem.proxyIP = Utils.String2List(TunProxyIP);
|
||||||
_config.tunModeItem.proxyProcess = Utils.String2List(TunProxyProcess);
|
_config.tunModeItem.proxyProcess = Utils.String2List(TunProxyProcess);
|
||||||
|
_config.tunModeItem.proxyDNS = Utils.ToJson(Utils.ParseJson(TunProxyDNS));
|
||||||
|
|
||||||
//coreType
|
//coreType
|
||||||
SaveCoreType();
|
SaveCoreType();
|
||||||
|
|||||||
@@ -299,7 +299,7 @@ namespace v2rayN.ViewModels
|
|||||||
UI.Show(ResUI.OperationSuccess);
|
UI.Show(ResUI.OperationSuccess);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
private void ImportRulesFromUrl()
|
private async Task ImportRulesFromUrl()
|
||||||
{
|
{
|
||||||
var url = SelectedRouting.url;
|
var url = SelectedRouting.url;
|
||||||
if (Utils.IsNullOrEmpty(url))
|
if (Utils.IsNullOrEmpty(url))
|
||||||
@@ -308,19 +308,16 @@ namespace v2rayN.ViewModels
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
Task.Run(async () =>
|
DownloadHandle downloadHandle = new DownloadHandle();
|
||||||
|
string result = await downloadHandle.TryDownloadString(url, true, "");
|
||||||
|
if (AddBatchRoutingRules(SelectedRouting, result) == 0)
|
||||||
{
|
{
|
||||||
DownloadHandle downloadHandle = new DownloadHandle();
|
Application.Current.Dispatcher.Invoke((Action)(() =>
|
||||||
string result = await downloadHandle.DownloadStringAsync(url, false, "");
|
|
||||||
if (AddBatchRoutingRules(SelectedRouting, result) == 0)
|
|
||||||
{
|
{
|
||||||
Application.Current.Dispatcher.Invoke((Action)(() =>
|
RefreshRulesItems();
|
||||||
{
|
}));
|
||||||
RefreshRulesItems();
|
UI.Show(ResUI.OperationSuccess);
|
||||||
}));
|
}
|
||||||
UI.Show(ResUI.OperationSuccess);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
private int AddBatchRoutingRules(RoutingItem routingItem, string clipboardData)
|
private int AddBatchRoutingRules(RoutingItem routingItem, string clipboardData)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -163,7 +163,7 @@ namespace v2rayN.ViewModels
|
|||||||
foreach (var item in routings)
|
foreach (var item in routings)
|
||||||
{
|
{
|
||||||
bool def = false;
|
bool def = false;
|
||||||
if (item.id.Equals(_config.routingBasicItem.routingIndexId))
|
if (item.id == _config.routingBasicItem.routingIndexId)
|
||||||
{
|
{
|
||||||
def = true;
|
def = true;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -137,12 +137,19 @@
|
|||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Style="{StaticResource ToolbarTextBlock}"
|
Style="{StaticResource ToolbarTextBlock}"
|
||||||
Text="{x:Static resx:ResUI.TbDisplayLog}" />
|
Text="{x:Static resx:ResUI.TbDisplayLog}" />
|
||||||
<ToggleButton
|
<StackPanel
|
||||||
x:Name="togDisplayLog"
|
|
||||||
Grid.Row="4"
|
Grid.Row="4"
|
||||||
Grid.Column="1"
|
Grid.Column="1"
|
||||||
Margin="4"
|
Margin="4"
|
||||||
HorizontalAlignment="Left" />
|
Orientation="Horizontal">
|
||||||
|
<ToggleButton x:Name="togDisplayLog" HorizontalAlignment="Left" />
|
||||||
|
<TextBlock
|
||||||
|
Margin="8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Style="{StaticResource ToolbarTextBlock}"
|
||||||
|
Text="{x:Static resx:ResUI.TipDisplayLog}" />
|
||||||
|
</StackPanel>
|
||||||
|
|
||||||
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Grid.Row="5"
|
Grid.Row="5"
|
||||||
@@ -176,7 +183,7 @@
|
|||||||
|
|
||||||
<Grid
|
<Grid
|
||||||
Grid.Row="1"
|
Grid.Row="1"
|
||||||
Margin="16"
|
Margin="8"
|
||||||
HorizontalAlignment="Center">
|
HorizontalAlignment="Center">
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
<ColumnDefinition Width="200" />
|
<ColumnDefinition Width="200" />
|
||||||
|
|||||||
@@ -10,8 +10,8 @@
|
|||||||
xmlns:resx="clr-namespace:v2rayN.Resx"
|
xmlns:resx="clr-namespace:v2rayN.Resx"
|
||||||
xmlns:vms="clr-namespace:v2rayN.ViewModels"
|
xmlns:vms="clr-namespace:v2rayN.ViewModels"
|
||||||
Title="{x:Static resx:ResUI.menuServers}"
|
Title="{x:Static resx:ResUI.menuServers}"
|
||||||
Width="800"
|
Width="820"
|
||||||
Height="800"
|
Height="820"
|
||||||
x:TypeArguments="vms:AddServerViewModel"
|
x:TypeArguments="vms:AddServerViewModel"
|
||||||
Background="{DynamicResource MaterialDesignPaper}"
|
Background="{DynamicResource MaterialDesignPaper}"
|
||||||
FontFamily="{x:Static conv:MaterialDesignFonts.MyFont}"
|
FontFamily="{x:Static conv:MaterialDesignFonts.MyFont}"
|
||||||
@@ -559,8 +559,10 @@
|
|||||||
Margin="{StaticResource ServerItemMargin}"
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
Style="{StaticResource DefComboBox}" />
|
Style="{StaticResource DefComboBox}" />
|
||||||
</Grid>
|
</Grid>
|
||||||
|
<Grid
|
||||||
<Grid x:Name="gridTlsMore" Grid.Row="7">
|
x:Name="gridTlsMore"
|
||||||
|
Grid.Row="7"
|
||||||
|
Visibility="Hidden">
|
||||||
<Grid.RowDefinitions>
|
<Grid.RowDefinitions>
|
||||||
<RowDefinition Height="Auto" />
|
<RowDefinition Height="Auto" />
|
||||||
<RowDefinition Height="Auto" />
|
<RowDefinition Height="Auto" />
|
||||||
@@ -595,25 +597,10 @@
|
|||||||
Margin="{StaticResource ServerItemMargin}"
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Style="{StaticResource ToolbarTextBlock}"
|
Style="{StaticResource ToolbarTextBlock}"
|
||||||
Text="{x:Static resx:ResUI.TbAllowInsecure}" />
|
|
||||||
<ComboBox
|
|
||||||
x:Name="cmbAllowInsecure"
|
|
||||||
Grid.Row="2"
|
|
||||||
Grid.Column="1"
|
|
||||||
Width="100"
|
|
||||||
Margin="{StaticResource ServerItemMargin}"
|
|
||||||
Style="{StaticResource DefComboBox}" />
|
|
||||||
|
|
||||||
<TextBlock
|
|
||||||
Grid.Row="3"
|
|
||||||
Grid.Column="0"
|
|
||||||
Margin="{StaticResource ServerItemMargin}"
|
|
||||||
VerticalAlignment="Center"
|
|
||||||
Style="{StaticResource ToolbarTextBlock}"
|
|
||||||
Text="{x:Static resx:ResUI.TbFingerprint}" />
|
Text="{x:Static resx:ResUI.TbFingerprint}" />
|
||||||
<ComboBox
|
<ComboBox
|
||||||
x:Name="cmbFingerprint"
|
x:Name="cmbFingerprint"
|
||||||
Grid.Row="3"
|
Grid.Row="2"
|
||||||
Grid.Column="1"
|
Grid.Column="1"
|
||||||
Width="200"
|
Width="200"
|
||||||
Margin="{StaticResource ServerItemMargin}"
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
@@ -621,7 +608,7 @@
|
|||||||
Style="{StaticResource DefComboBox}" />
|
Style="{StaticResource DefComboBox}" />
|
||||||
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Grid.Row="4"
|
Grid.Row="3"
|
||||||
Grid.Column="0"
|
Grid.Column="0"
|
||||||
Margin="{StaticResource ServerItemMargin}"
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
@@ -629,13 +616,125 @@
|
|||||||
Text="{x:Static resx:ResUI.TbAlpn}" />
|
Text="{x:Static resx:ResUI.TbAlpn}" />
|
||||||
<ComboBox
|
<ComboBox
|
||||||
x:Name="cmbAlpn"
|
x:Name="cmbAlpn"
|
||||||
Grid.Row="4"
|
Grid.Row="3"
|
||||||
Grid.Column="1"
|
Grid.Column="1"
|
||||||
Width="200"
|
Width="200"
|
||||||
Margin="{StaticResource ServerItemMargin}"
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
Style="{StaticResource DefComboBox}" />
|
Style="{StaticResource DefComboBox}" />
|
||||||
</Grid>
|
|
||||||
|
|
||||||
|
<TextBlock
|
||||||
|
Grid.Row="4"
|
||||||
|
Grid.Column="0"
|
||||||
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Style="{StaticResource ToolbarTextBlock}"
|
||||||
|
Text="{x:Static resx:ResUI.TbAllowInsecure}" />
|
||||||
|
<ComboBox
|
||||||
|
x:Name="cmbAllowInsecure"
|
||||||
|
Grid.Row="4"
|
||||||
|
Grid.Column="1"
|
||||||
|
Width="100"
|
||||||
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
|
Style="{StaticResource DefComboBox}" />
|
||||||
|
</Grid>
|
||||||
|
<Grid
|
||||||
|
x:Name="gridRealityMore"
|
||||||
|
Grid.Row="7"
|
||||||
|
Visibility="Hidden">
|
||||||
|
<Grid.RowDefinitions>
|
||||||
|
<RowDefinition Height="Auto" />
|
||||||
|
<RowDefinition Height="Auto" />
|
||||||
|
<RowDefinition Height="Auto" />
|
||||||
|
<RowDefinition Height="Auto" />
|
||||||
|
<RowDefinition Height="Auto" />
|
||||||
|
</Grid.RowDefinitions>
|
||||||
|
<Grid.ColumnDefinitions>
|
||||||
|
<ColumnDefinition Width="180" />
|
||||||
|
<ColumnDefinition Width="Auto" />
|
||||||
|
</Grid.ColumnDefinitions>
|
||||||
|
|
||||||
|
<TextBlock
|
||||||
|
Grid.Row="0"
|
||||||
|
Grid.Column="0"
|
||||||
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Style="{StaticResource ToolbarTextBlock}"
|
||||||
|
Text="{x:Static resx:ResUI.TbSNI}" />
|
||||||
|
<TextBox
|
||||||
|
x:Name="txtSNI2"
|
||||||
|
Grid.Row="0"
|
||||||
|
Grid.Column="1"
|
||||||
|
Width="400"
|
||||||
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
|
HorizontalAlignment="Left"
|
||||||
|
Style="{StaticResource DefTextBox}" />
|
||||||
|
|
||||||
|
<TextBlock
|
||||||
|
Grid.Row="1"
|
||||||
|
Grid.Column="0"
|
||||||
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Style="{StaticResource ToolbarTextBlock}"
|
||||||
|
Text="{x:Static resx:ResUI.TbFingerprint}" />
|
||||||
|
<ComboBox
|
||||||
|
x:Name="cmbFingerprint2"
|
||||||
|
Grid.Row="1"
|
||||||
|
Grid.Column="1"
|
||||||
|
Width="200"
|
||||||
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
|
IsEditable="True"
|
||||||
|
Style="{StaticResource DefComboBox}" />
|
||||||
|
|
||||||
|
<TextBlock
|
||||||
|
Grid.Row="2"
|
||||||
|
Grid.Column="0"
|
||||||
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Style="{StaticResource ToolbarTextBlock}"
|
||||||
|
Text="{x:Static resx:ResUI.TbPublicKey}" />
|
||||||
|
<TextBox
|
||||||
|
x:Name="txtPublicKey"
|
||||||
|
Grid.Row="2"
|
||||||
|
Grid.Column="1"
|
||||||
|
Width="400"
|
||||||
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
|
HorizontalAlignment="Left"
|
||||||
|
Style="{StaticResource DefTextBox}" />
|
||||||
|
|
||||||
|
<TextBlock
|
||||||
|
Grid.Row="3"
|
||||||
|
Grid.Column="0"
|
||||||
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Style="{StaticResource ToolbarTextBlock}"
|
||||||
|
Text="{x:Static resx:ResUI.TbShortId}" />
|
||||||
|
<TextBox
|
||||||
|
x:Name="txtShortId"
|
||||||
|
Grid.Row="3"
|
||||||
|
Grid.Column="1"
|
||||||
|
Width="200"
|
||||||
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
|
HorizontalAlignment="Left"
|
||||||
|
Style="{StaticResource DefTextBox}" />
|
||||||
|
|
||||||
|
<TextBlock
|
||||||
|
Grid.Row="4"
|
||||||
|
Grid.Column="0"
|
||||||
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Style="{StaticResource ToolbarTextBlock}"
|
||||||
|
Text="{x:Static resx:ResUI.TbSpiderX}" />
|
||||||
|
<TextBox
|
||||||
|
x:Name="txtSpiderX"
|
||||||
|
Grid.Row="4"
|
||||||
|
Grid.Column="1"
|
||||||
|
Width="400"
|
||||||
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
|
HorizontalAlignment="Left"
|
||||||
|
Style="{StaticResource DefTextBox}" />
|
||||||
|
|
||||||
|
|
||||||
|
</Grid>
|
||||||
<Separator
|
<Separator
|
||||||
Grid.Row="8"
|
Grid.Row="8"
|
||||||
Margin="0,2"
|
Margin="0,2"
|
||||||
@@ -643,7 +742,7 @@
|
|||||||
|
|
||||||
<Grid
|
<Grid
|
||||||
Grid.Row="9"
|
Grid.Row="9"
|
||||||
Margin="16"
|
Margin="8"
|
||||||
HorizontalAlignment="Center">
|
HorizontalAlignment="Center">
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
<ColumnDefinition Width="200" />
|
<ColumnDefinition Width="200" />
|
||||||
|
|||||||
@@ -39,6 +39,7 @@ namespace v2rayN.Views
|
|||||||
Global.fingerprints.ForEach(it =>
|
Global.fingerprints.ForEach(it =>
|
||||||
{
|
{
|
||||||
cmbFingerprint.Items.Add(it);
|
cmbFingerprint.Items.Add(it);
|
||||||
|
cmbFingerprint2.Items.Add(it);
|
||||||
});
|
});
|
||||||
Global.allowInsecures.ForEach(it =>
|
Global.allowInsecures.ForEach(it =>
|
||||||
{
|
{
|
||||||
@@ -74,8 +75,8 @@ namespace v2rayN.Views
|
|||||||
break;
|
break;
|
||||||
case EConfigType.VLESS:
|
case EConfigType.VLESS:
|
||||||
gridVLESS.Visibility = Visibility.Visible;
|
gridVLESS.Visibility = Visibility.Visible;
|
||||||
cmbStreamSecurity.Items.Add(Global.StreamSecurityX);
|
cmbStreamSecurity.Items.Add(Global.StreamSecurityReality);
|
||||||
Global.xtlsFlows.ForEach(it =>
|
Global.flows.ForEach(it =>
|
||||||
{
|
{
|
||||||
cmbFlow5.Items.Add(it);
|
cmbFlow5.Items.Add(it);
|
||||||
});
|
});
|
||||||
@@ -86,8 +87,7 @@ namespace v2rayN.Views
|
|||||||
break;
|
break;
|
||||||
case EConfigType.Trojan:
|
case EConfigType.Trojan:
|
||||||
gridTrojan.Visibility = Visibility.Visible;
|
gridTrojan.Visibility = Visibility.Visible;
|
||||||
cmbStreamSecurity.Items.Add(Global.StreamSecurityX);
|
Global.flows.ForEach(it =>
|
||||||
Global.xtlsFlows.ForEach(it =>
|
|
||||||
{
|
{
|
||||||
cmbFlow6.Items.Add(it);
|
cmbFlow6.Items.Add(it);
|
||||||
});
|
});
|
||||||
@@ -138,6 +138,13 @@ namespace v2rayN.Views
|
|||||||
this.Bind(ViewModel, vm => vm.SelectedSource.allowInsecure, v => v.cmbAllowInsecure.Text).DisposeWith(disposables);
|
this.Bind(ViewModel, vm => vm.SelectedSource.allowInsecure, v => v.cmbAllowInsecure.Text).DisposeWith(disposables);
|
||||||
this.Bind(ViewModel, vm => vm.SelectedSource.fingerprint, v => v.cmbFingerprint.Text).DisposeWith(disposables);
|
this.Bind(ViewModel, vm => vm.SelectedSource.fingerprint, v => v.cmbFingerprint.Text).DisposeWith(disposables);
|
||||||
this.Bind(ViewModel, vm => vm.SelectedSource.alpn, v => v.cmbAlpn.Text).DisposeWith(disposables);
|
this.Bind(ViewModel, vm => vm.SelectedSource.alpn, v => v.cmbAlpn.Text).DisposeWith(disposables);
|
||||||
|
//reality
|
||||||
|
this.Bind(ViewModel, vm => vm.SelectedSource.sni, v => v.txtSNI2.Text).DisposeWith(disposables);
|
||||||
|
this.Bind(ViewModel, vm => vm.SelectedSource.fingerprint, v => v.cmbFingerprint2.Text).DisposeWith(disposables);
|
||||||
|
this.Bind(ViewModel, vm => vm.SelectedSource.publicKey, v => v.txtPublicKey.Text).DisposeWith(disposables);
|
||||||
|
this.Bind(ViewModel, vm => vm.SelectedSource.shortId, v => v.txtShortId.Text).DisposeWith(disposables);
|
||||||
|
this.Bind(ViewModel, vm => vm.SelectedSource.spiderX, v => v.txtSpiderX.Text).DisposeWith(disposables);
|
||||||
|
|
||||||
|
|
||||||
this.BindCommand(ViewModel, vm => vm.SaveCmd, v => v.btnSave).DisposeWith(disposables);
|
this.BindCommand(ViewModel, vm => vm.SaveCmd, v => v.btnSave).DisposeWith(disposables);
|
||||||
|
|
||||||
@@ -159,13 +166,20 @@ namespace v2rayN.Views
|
|||||||
private void CmbStreamSecurity_SelectionChanged(object sender, SelectionChangedEventArgs e)
|
private void CmbStreamSecurity_SelectionChanged(object sender, SelectionChangedEventArgs e)
|
||||||
{
|
{
|
||||||
var security = cmbStreamSecurity.SelectedItem.ToString();
|
var security = cmbStreamSecurity.SelectedItem.ToString();
|
||||||
if (Utils.IsNullOrEmpty(security))
|
if (security == Global.StreamSecurityReality)
|
||||||
{
|
{
|
||||||
|
gridRealityMore.Visibility = Visibility.Visible;
|
||||||
gridTlsMore.Visibility = Visibility.Hidden;
|
gridTlsMore.Visibility = Visibility.Hidden;
|
||||||
}
|
}
|
||||||
|
else if (security == Global.StreamSecurity)
|
||||||
|
{
|
||||||
|
gridRealityMore.Visibility = Visibility.Hidden;
|
||||||
|
gridTlsMore.Visibility = Visibility.Visible;
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
gridTlsMore.Visibility = Visibility.Visible;
|
gridRealityMore.Visibility = Visibility.Hidden;
|
||||||
|
gridTlsMore.Visibility = Visibility.Hidden;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
private void btnGUID_Click(object sender, RoutedEventArgs e)
|
private void btnGUID_Click(object sender, RoutedEventArgs e)
|
||||||
@@ -185,12 +199,12 @@ namespace v2rayN.Views
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (network.Equals(Global.DefaultNetwork))
|
if (network == Global.DefaultNetwork)
|
||||||
{
|
{
|
||||||
cmbHeaderType.Items.Add(Global.None);
|
cmbHeaderType.Items.Add(Global.None);
|
||||||
cmbHeaderType.Items.Add(Global.TcpHeaderHttp);
|
cmbHeaderType.Items.Add(Global.TcpHeaderHttp);
|
||||||
}
|
}
|
||||||
else if (network.Equals("kcp") || network.Equals("quic"))
|
else if (network is "kcp" or "quic")
|
||||||
{
|
{
|
||||||
cmbHeaderType.Items.Add(Global.None);
|
cmbHeaderType.Items.Add(Global.None);
|
||||||
Global.kcpHeaderTypes.ForEach(it =>
|
Global.kcpHeaderTypes.ForEach(it =>
|
||||||
@@ -198,7 +212,7 @@ namespace v2rayN.Views
|
|||||||
cmbHeaderType.Items.Add(it);
|
cmbHeaderType.Items.Add(it);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
else if (network.Equals("grpc"))
|
else if (network == "grpc")
|
||||||
{
|
{
|
||||||
cmbHeaderType.Items.Add(Global.GrpcgunMode);
|
cmbHeaderType.Items.Add(Global.GrpcgunMode);
|
||||||
cmbHeaderType.Items.Add(Global.GrpcmultiMode);
|
cmbHeaderType.Items.Add(Global.GrpcmultiMode);
|
||||||
@@ -222,37 +236,34 @@ namespace v2rayN.Views
|
|||||||
tipPath.Text =
|
tipPath.Text =
|
||||||
tipHeaderType.Text = string.Empty;
|
tipHeaderType.Text = string.Empty;
|
||||||
|
|
||||||
if (network.Equals(Global.DefaultNetwork))
|
switch (network)
|
||||||
{
|
{
|
||||||
tipRequestHost.Text = ResUI.TransportRequestHostTip1;
|
case Global.DefaultNetwork:
|
||||||
tipHeaderType.Text = ResUI.TransportHeaderTypeTip1;
|
tipRequestHost.Text = ResUI.TransportRequestHostTip1;
|
||||||
}
|
tipHeaderType.Text = ResUI.TransportHeaderTypeTip1;
|
||||||
else if (network.Equals("kcp"))
|
break;
|
||||||
{
|
case "kcp":
|
||||||
tipHeaderType.Text = ResUI.TransportHeaderTypeTip2;
|
tipHeaderType.Text = ResUI.TransportHeaderTypeTip2;
|
||||||
tipPath.Text = ResUI.TransportPathTip5;
|
tipPath.Text = ResUI.TransportPathTip5;
|
||||||
}
|
break;
|
||||||
else if (network.Equals("ws"))
|
case "ws":
|
||||||
{
|
tipRequestHost.Text = ResUI.TransportRequestHostTip2;
|
||||||
tipRequestHost.Text = ResUI.TransportRequestHostTip2;
|
tipPath.Text = ResUI.TransportPathTip1;
|
||||||
tipPath.Text = ResUI.TransportPathTip1;
|
break;
|
||||||
}
|
case "h2":
|
||||||
else if (network.Equals("h2"))
|
tipRequestHost.Text = ResUI.TransportRequestHostTip3;
|
||||||
{
|
tipPath.Text = ResUI.TransportPathTip2;
|
||||||
tipRequestHost.Text = ResUI.TransportRequestHostTip3;
|
break;
|
||||||
tipPath.Text = ResUI.TransportPathTip2;
|
case "quic":
|
||||||
}
|
tipRequestHost.Text = ResUI.TransportRequestHostTip4;
|
||||||
else if (network.Equals("quic"))
|
tipPath.Text = ResUI.TransportPathTip3;
|
||||||
{
|
tipHeaderType.Text = ResUI.TransportHeaderTypeTip3;
|
||||||
tipRequestHost.Text = ResUI.TransportRequestHostTip4;
|
break;
|
||||||
tipPath.Text = ResUI.TransportPathTip3;
|
case "grpc":
|
||||||
tipHeaderType.Text = ResUI.TransportHeaderTypeTip3;
|
tipPath.Text = ResUI.TransportPathTip4;
|
||||||
}
|
tipHeaderType.Text = ResUI.TransportHeaderTypeTip4;
|
||||||
else if (network.Equals("grpc"))
|
labHeaderType.Visibility = Visibility.Hidden;
|
||||||
{
|
break;
|
||||||
tipPath.Text = ResUI.TransportPathTip4;
|
|
||||||
tipHeaderType.Text = ResUI.TransportHeaderTypeTip4;
|
|
||||||
labHeaderType.Visibility = Visibility.Hidden;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -150,7 +150,7 @@
|
|||||||
|
|
||||||
<Grid
|
<Grid
|
||||||
Grid.Row="2"
|
Grid.Row="2"
|
||||||
Margin="16"
|
Margin="8"
|
||||||
HorizontalAlignment="Center">
|
HorizontalAlignment="Center">
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
<ColumnDefinition Width="150" />
|
<ColumnDefinition Width="150" />
|
||||||
|
|||||||
@@ -1,47 +1,24 @@
|
|||||||
using System.Windows;
|
using System.Text;
|
||||||
|
using System.Windows;
|
||||||
using System.Windows.Controls;
|
using System.Windows.Controls;
|
||||||
using System.Windows.Input;
|
using System.Windows.Input;
|
||||||
using v2rayN.Handler;
|
using v2rayN.Handler;
|
||||||
using v2rayN.Mode;
|
using v2rayN.Mode;
|
||||||
using v2rayN.Resx;
|
using v2rayN.Resx;
|
||||||
using Forms = System.Windows.Forms;
|
|
||||||
|
|
||||||
namespace v2rayN.Views
|
namespace v2rayN.Views
|
||||||
{
|
{
|
||||||
public partial class GlobalHotkeySettingWindow
|
public partial class GlobalHotkeySettingWindow
|
||||||
{
|
{
|
||||||
private static Config _config;
|
private static Config _config = default!;
|
||||||
List<KeyEventItem> lstKey;
|
private Dictionary<object, KeyEventItem> _TextBoxKeyEventItem = default!;
|
||||||
|
|
||||||
public GlobalHotkeySettingWindow()
|
public GlobalHotkeySettingWindow()
|
||||||
{
|
{
|
||||||
InitializeComponent();
|
InitializeComponent();
|
||||||
this.Owner = Application.Current.MainWindow;
|
this.Owner = Application.Current.MainWindow;
|
||||||
_config = LazyConfig.Instance.GetConfig();
|
_config = LazyConfig.Instance.GetConfig();
|
||||||
|
_config.globalHotkeys ??= new List<KeyEventItem>();
|
||||||
if (_config.globalHotkeys == null)
|
|
||||||
{
|
|
||||||
_config.globalHotkeys = new List<KeyEventItem>();
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (EGlobalHotkey it in Enum.GetValues(typeof(EGlobalHotkey)))
|
|
||||||
{
|
|
||||||
if (_config.globalHotkeys.FindIndex(t => t.eGlobalHotkey == it) >= 0)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
_config.globalHotkeys.Add(new KeyEventItem()
|
|
||||||
{
|
|
||||||
eGlobalHotkey = it,
|
|
||||||
Alt = false,
|
|
||||||
Control = false,
|
|
||||||
Shift = false,
|
|
||||||
KeyCode = null
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
lstKey = Utils.DeepCopy(_config.globalHotkeys);
|
|
||||||
|
|
||||||
txtGlobalHotkey0.KeyDown += TxtGlobalHotkey_KeyDown;
|
txtGlobalHotkey0.KeyDown += TxtGlobalHotkey_KeyDown;
|
||||||
txtGlobalHotkey1.KeyDown += TxtGlobalHotkey_KeyDown;
|
txtGlobalHotkey1.KeyDown += TxtGlobalHotkey_KeyDown;
|
||||||
@@ -49,67 +26,84 @@ namespace v2rayN.Views
|
|||||||
txtGlobalHotkey3.KeyDown += TxtGlobalHotkey_KeyDown;
|
txtGlobalHotkey3.KeyDown += TxtGlobalHotkey_KeyDown;
|
||||||
txtGlobalHotkey4.KeyDown += TxtGlobalHotkey_KeyDown;
|
txtGlobalHotkey4.KeyDown += TxtGlobalHotkey_KeyDown;
|
||||||
|
|
||||||
BindingData(-1);
|
HotkeyHandler.Instance.IsPause = true;
|
||||||
|
this.Closing += (s, e) => HotkeyHandler.Instance.IsPause = false;
|
||||||
Utils.SetDarkBorder(this, _config.uiItem.colorModeDark);
|
Utils.SetDarkBorder(this, _config.uiItem.colorModeDark);
|
||||||
|
InitData();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void InitData()
|
||||||
|
{
|
||||||
|
_TextBoxKeyEventItem = new()
|
||||||
|
{
|
||||||
|
{ txtGlobalHotkey0,GetKeyEventItemByEGlobalHotkey(_config.globalHotkeys,EGlobalHotkey.ShowForm) },
|
||||||
|
{ txtGlobalHotkey1,GetKeyEventItemByEGlobalHotkey(_config.globalHotkeys,EGlobalHotkey.SystemProxyClear) },
|
||||||
|
{ txtGlobalHotkey2,GetKeyEventItemByEGlobalHotkey(_config.globalHotkeys,EGlobalHotkey.SystemProxySet) },
|
||||||
|
{ txtGlobalHotkey3,GetKeyEventItemByEGlobalHotkey(_config.globalHotkeys,EGlobalHotkey.SystemProxyUnchanged)},
|
||||||
|
{ txtGlobalHotkey4,GetKeyEventItemByEGlobalHotkey(_config.globalHotkeys,EGlobalHotkey.SystemProxyPac)}
|
||||||
|
};
|
||||||
|
BindingData();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private void TxtGlobalHotkey_KeyDown(object sender, KeyEventArgs e)
|
private void TxtGlobalHotkey_KeyDown(object sender, KeyEventArgs e)
|
||||||
{
|
{
|
||||||
var txt = ((TextBox)sender);
|
e.Handled = true;
|
||||||
var index = Utils.ToInt(txt.Name.Substring(txt.Name.Length - 1, 1));
|
var _ModifierKeys = new Key[] { Key.LeftCtrl, Key.RightCtrl, Key.LeftShift,
|
||||||
|
Key.RightShift, Key.LeftAlt, Key.RightAlt, Key.LWin, Key.RWin};
|
||||||
if (e.Key == Key.System)
|
_TextBoxKeyEventItem[sender].KeyCode = e.Key == Key.System ? (_ModifierKeys.Contains(e.SystemKey) ? Key.None : e.SystemKey) : (_ModifierKeys.Contains(e.Key) ? Key.None : e.Key);
|
||||||
return;
|
_TextBoxKeyEventItem[sender].Alt = (Keyboard.Modifiers & ModifierKeys.Alt) == ModifierKeys.Alt;
|
||||||
var formsKey = (Forms.Keys)KeyInterop.VirtualKeyFromKey(e.Key);
|
_TextBoxKeyEventItem[sender].Control = (Keyboard.Modifiers & ModifierKeys.Control) == ModifierKeys.Control;
|
||||||
|
_TextBoxKeyEventItem[sender].Shift = (Keyboard.Modifiers & ModifierKeys.Shift) == ModifierKeys.Shift;
|
||||||
lstKey[index].KeyCode = formsKey;
|
(sender as TextBox)!.Text = KeyEventItemToString(_TextBoxKeyEventItem[sender]);
|
||||||
lstKey[index].Alt = Keyboard.IsKeyDown(Key.LeftAlt) || Keyboard.IsKeyDown(Key.RightAlt);
|
|
||||||
lstKey[index].Control = Keyboard.IsKeyDown(Key.LeftCtrl) || Keyboard.IsKeyDown(Key.RightCtrl);
|
|
||||||
lstKey[index].Shift = Keyboard.IsKeyDown(Key.LeftShift) || Keyboard.IsKeyDown(Key.RightShift);
|
|
||||||
|
|
||||||
BindingData(index);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void BindingData(int index)
|
private KeyEventItem GetKeyEventItemByEGlobalHotkey(List<KeyEventItem> KELsit, EGlobalHotkey eg)
|
||||||
{
|
{
|
||||||
for (int k = 0; k < lstKey.Count; k++)
|
return Utils.DeepCopy(KELsit.Find((it) => it.eGlobalHotkey == eg) ?? new()
|
||||||
{
|
{
|
||||||
if (index >= 0 && index != k)
|
eGlobalHotkey = eg,
|
||||||
{
|
Control = false,
|
||||||
continue;
|
Alt = false,
|
||||||
}
|
Shift = false,
|
||||||
var item = lstKey[k];
|
KeyCode = null
|
||||||
var keys = string.Empty;
|
});
|
||||||
|
|
||||||
if (item.Control)
|
}
|
||||||
{
|
private string KeyEventItemToString(KeyEventItem item)
|
||||||
keys += $"{Forms.Keys.Control.ToString()} + ";
|
{
|
||||||
}
|
var res = new StringBuilder();
|
||||||
if (item.Alt)
|
|
||||||
{
|
|
||||||
keys += $"{Forms.Keys.Alt.ToString()} + ";
|
|
||||||
}
|
|
||||||
if (item.Shift)
|
|
||||||
{
|
|
||||||
keys += $"{Forms.Keys.Shift.ToString()} + ";
|
|
||||||
}
|
|
||||||
if (item.KeyCode != null)
|
|
||||||
{
|
|
||||||
keys += $"{item.KeyCode.ToString()}";
|
|
||||||
}
|
|
||||||
|
|
||||||
SetText($"txtGlobalHotkey{k}", keys);
|
if (item.Control) res.Append($"{ModifierKeys.Control}+");
|
||||||
|
if (item.Shift) res.Append($"{ModifierKeys.Shift}+");
|
||||||
|
if (item.Alt) res.Append($"{ModifierKeys.Alt}+");
|
||||||
|
if (item.KeyCode != null && item.KeyCode != Key.None)
|
||||||
|
res.Append($"{item.KeyCode}");
|
||||||
|
|
||||||
|
return res.ToString();
|
||||||
|
}
|
||||||
|
private void BindingData()
|
||||||
|
{
|
||||||
|
foreach (var item in _TextBoxKeyEventItem)
|
||||||
|
{
|
||||||
|
if (item.Value.KeyCode != null && item.Value.KeyCode != Key.None)
|
||||||
|
{
|
||||||
|
(item.Key as TextBox)!.Text = KeyEventItemToString(item.Value);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
(item.Key as TextBox)!.Text = string.Empty;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void btnSave_Click(object sender, RoutedEventArgs e)
|
private void btnSave_Click(object sender, RoutedEventArgs e)
|
||||||
{
|
{
|
||||||
_config.globalHotkeys = lstKey;
|
_config.globalHotkeys = _TextBoxKeyEventItem.Values.ToList();
|
||||||
|
|
||||||
if (ConfigHandler.SaveConfig(ref _config, false) == 0)
|
if (ConfigHandler.SaveConfig(ref _config, false) == 0)
|
||||||
{
|
{
|
||||||
|
HotkeyHandler.Instance.ReLoad();
|
||||||
this.DialogResult = true;
|
this.DialogResult = true;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -125,37 +119,14 @@ namespace v2rayN.Views
|
|||||||
|
|
||||||
private void btnReset_Click(object sender, RoutedEventArgs e)
|
private void btnReset_Click(object sender, RoutedEventArgs e)
|
||||||
{
|
{
|
||||||
lstKey.Clear();
|
foreach (var k in _TextBoxKeyEventItem.Keys)
|
||||||
foreach (EGlobalHotkey it in Enum.GetValues(typeof(EGlobalHotkey)))
|
|
||||||
{
|
{
|
||||||
if (lstKey.FindIndex(t => t.eGlobalHotkey == it) >= 0)
|
_TextBoxKeyEventItem[k].Alt = false;
|
||||||
{
|
_TextBoxKeyEventItem[k].Control = false;
|
||||||
continue;
|
_TextBoxKeyEventItem[k].Shift = false;
|
||||||
}
|
_TextBoxKeyEventItem[k].KeyCode = Key.None;
|
||||||
|
|
||||||
lstKey.Add(new KeyEventItem()
|
|
||||||
{
|
|
||||||
eGlobalHotkey = it,
|
|
||||||
Alt = false,
|
|
||||||
Control = false,
|
|
||||||
Shift = false,
|
|
||||||
KeyCode = null
|
|
||||||
});
|
|
||||||
}
|
|
||||||
BindingData(-1);
|
|
||||||
}
|
|
||||||
private void SetText(string name, string txt)
|
|
||||||
{
|
|
||||||
foreach (UIElement element in gridText.Children)
|
|
||||||
{
|
|
||||||
if (element is TextBox)
|
|
||||||
{
|
|
||||||
if (((TextBox)element).Name == name)
|
|
||||||
{
|
|
||||||
((TextBox)element).Text = txt;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
BindingData();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void GlobalHotkeySettingWindow_KeyDown(object sender, KeyEventArgs e)
|
private void GlobalHotkeySettingWindow_KeyDown(object sender, KeyEventArgs e)
|
||||||
|
|||||||
@@ -15,8 +15,8 @@
|
|||||||
Title="v2rayN"
|
Title="v2rayN"
|
||||||
Width="900"
|
Width="900"
|
||||||
Height="700"
|
Height="700"
|
||||||
MinWidth="800"
|
MinWidth="900"
|
||||||
MinHeight="600"
|
MinHeight="700"
|
||||||
x:TypeArguments="vms:MainWindowViewModel"
|
x:TypeArguments="vms:MainWindowViewModel"
|
||||||
Background="{DynamicResource MaterialDesignPaper}"
|
Background="{DynamicResource MaterialDesignPaper}"
|
||||||
FontFamily="{x:Static conv:MaterialDesignFonts.MyFont}"
|
FontFamily="{x:Static conv:MaterialDesignFonts.MyFont}"
|
||||||
@@ -53,7 +53,10 @@
|
|||||||
<MenuItem Padding="8,0">
|
<MenuItem Padding="8,0">
|
||||||
<MenuItem.Header>
|
<MenuItem.Header>
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="Server" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="Server" />
|
||||||
<TextBlock Text="{x:Static resx:ResUI.menuServers}" />
|
<TextBlock Text="{x:Static resx:ResUI.menuServers}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</MenuItem.Header>
|
</MenuItem.Header>
|
||||||
@@ -99,7 +102,10 @@
|
|||||||
<MenuItem Padding="8,0">
|
<MenuItem Padding="8,0">
|
||||||
<MenuItem.Header>
|
<MenuItem.Header>
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="BookClockOutline" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="BookClockOutline" />
|
||||||
<TextBlock Text="{x:Static resx:ResUI.menuSubscription}" />
|
<TextBlock Text="{x:Static resx:ResUI.menuSubscription}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</MenuItem.Header>
|
</MenuItem.Header>
|
||||||
@@ -131,7 +137,10 @@
|
|||||||
<MenuItem Padding="8,0">
|
<MenuItem Padding="8,0">
|
||||||
<MenuItem.Header>
|
<MenuItem.Header>
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="SettingsOutline" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="SettingsOutline" />
|
||||||
<TextBlock Text="{x:Static resx:ResUI.menuSetting}" />
|
<TextBlock Text="{x:Static resx:ResUI.menuSetting}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</MenuItem.Header>
|
</MenuItem.Header>
|
||||||
@@ -169,7 +178,10 @@
|
|||||||
<MenuItem x:Name="menuReload" Padding="8,0">
|
<MenuItem x:Name="menuReload" Padding="8,0">
|
||||||
<MenuItem.Header>
|
<MenuItem.Header>
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="Reload" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="Reload" />
|
||||||
<TextBlock Text="{x:Static resx:ResUI.menuReload}" />
|
<TextBlock Text="{x:Static resx:ResUI.menuReload}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</MenuItem.Header>
|
</MenuItem.Header>
|
||||||
@@ -180,7 +192,10 @@
|
|||||||
<MenuItem Padding="8,0">
|
<MenuItem Padding="8,0">
|
||||||
<MenuItem.Header>
|
<MenuItem.Header>
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="Update" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="Update" />
|
||||||
<TextBlock Text="{x:Static resx:ResUI.menuCheckUpdate}" />
|
<TextBlock Text="{x:Static resx:ResUI.menuCheckUpdate}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</MenuItem.Header>
|
</MenuItem.Header>
|
||||||
@@ -221,7 +236,10 @@
|
|||||||
<MenuItem x:Name="menuHelp" Padding="8,0">
|
<MenuItem x:Name="menuHelp" Padding="8,0">
|
||||||
<MenuItem.Header>
|
<MenuItem.Header>
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="HelpCircleOutline" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="HelpCircleOutline" />
|
||||||
<TextBlock Text="{x:Static resx:ResUI.menuHelp}" />
|
<TextBlock Text="{x:Static resx:ResUI.menuHelp}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</MenuItem.Header>
|
</MenuItem.Header>
|
||||||
@@ -235,7 +253,10 @@
|
|||||||
Click="menuPromotion_Click">
|
Click="menuPromotion_Click">
|
||||||
<MenuItem.Header>
|
<MenuItem.Header>
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="VolumeHigh" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="VolumeHigh" />
|
||||||
<TextBlock Text="{x:Static resx:ResUI.menuPromotion}" />
|
<TextBlock Text="{x:Static resx:ResUI.menuPromotion}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</MenuItem.Header>
|
</MenuItem.Header>
|
||||||
@@ -249,7 +270,10 @@
|
|||||||
Click="menuClose_Click">
|
Click="menuClose_Click">
|
||||||
<MenuItem.Header>
|
<MenuItem.Header>
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="Minimize" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="Minimize" />
|
||||||
<TextBlock Text="{x:Static resx:ResUI.menuClose}" />
|
<TextBlock Text="{x:Static resx:ResUI.menuClose}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</MenuItem.Header>
|
</MenuItem.Header>
|
||||||
@@ -259,6 +283,7 @@
|
|||||||
<materialDesign:PopupBox
|
<materialDesign:PopupBox
|
||||||
Padding="8,0"
|
Padding="8,0"
|
||||||
HorizontalAlignment="Right"
|
HorizontalAlignment="Right"
|
||||||
|
StaysOpen="True"
|
||||||
Style="{StaticResource MaterialDesignToolForegroundPopupBox}">
|
Style="{StaticResource MaterialDesignToolForegroundPopupBox}">
|
||||||
<StackPanel Margin="8">
|
<StackPanel Margin="8">
|
||||||
<Grid>
|
<Grid>
|
||||||
@@ -351,12 +376,13 @@
|
|||||||
Height="30"
|
Height="30"
|
||||||
Margin="4,0"
|
Margin="4,0"
|
||||||
Style="{StaticResource MaterialDesignFloatingActionMiniLightButton}">
|
Style="{StaticResource MaterialDesignFloatingActionMiniLightButton}">
|
||||||
<materialDesign:PackIcon Kind="Plus" />
|
<materialDesign:PackIcon VerticalAlignment="Center" Kind="Plus" />
|
||||||
</Button>
|
</Button>
|
||||||
<TextBox
|
<TextBox
|
||||||
x:Name="txtServerFilter"
|
x:Name="txtServerFilter"
|
||||||
Width="200"
|
Width="200"
|
||||||
Margin="4,0"
|
Margin="4,0"
|
||||||
|
VerticalContentAlignment="Center"
|
||||||
materialDesign:HintAssist.Hint="{x:Static resx:ResUI.MsgServerTitle}"
|
materialDesign:HintAssist.Hint="{x:Static resx:ResUI.MsgServerTitle}"
|
||||||
materialDesign:TextFieldAssist.HasClearButton="True"
|
materialDesign:TextFieldAssist.HasClearButton="True"
|
||||||
Style="{StaticResource DefTextBox}" />
|
Style="{StaticResource DefTextBox}" />
|
||||||
@@ -458,6 +484,7 @@
|
|||||||
CanUserSortColumns="False"
|
CanUserSortColumns="False"
|
||||||
EnableRowVirtualization="True"
|
EnableRowVirtualization="True"
|
||||||
Focusable="True"
|
Focusable="True"
|
||||||
|
FrozenColumnCount="1"
|
||||||
GridLinesVisibility="All"
|
GridLinesVisibility="All"
|
||||||
HeadersVisibility="All"
|
HeadersVisibility="All"
|
||||||
IsReadOnly="True"
|
IsReadOnly="True"
|
||||||
@@ -538,22 +565,24 @@
|
|||||||
</MenuItem.Header>
|
</MenuItem.Header>
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
<MenuItem
|
<MenuItem Header="{x:Static resx:ResUI.menuMoveTo}">
|
||||||
x:Name="menuMoveTop"
|
<MenuItem
|
||||||
Height="{StaticResource MenuItemHeight}"
|
x:Name="menuMoveTop"
|
||||||
Header="{x:Static resx:ResUI.menuMoveTop}" />
|
Height="{StaticResource MenuItemHeight}"
|
||||||
<MenuItem
|
Header="{x:Static resx:ResUI.menuMoveTop}" />
|
||||||
x:Name="menuMoveUp"
|
<MenuItem
|
||||||
Height="{StaticResource MenuItemHeight}"
|
x:Name="menuMoveUp"
|
||||||
Header="{x:Static resx:ResUI.menuMoveUp}" />
|
Height="{StaticResource MenuItemHeight}"
|
||||||
<MenuItem
|
Header="{x:Static resx:ResUI.menuMoveUp}" />
|
||||||
x:Name="menuMoveDown"
|
<MenuItem
|
||||||
Height="{StaticResource MenuItemHeight}"
|
x:Name="menuMoveDown"
|
||||||
Header="{x:Static resx:ResUI.menuMoveDown}" />
|
Height="{StaticResource MenuItemHeight}"
|
||||||
<MenuItem
|
Header="{x:Static resx:ResUI.menuMoveDown}" />
|
||||||
x:Name="menuMoveBottom"
|
<MenuItem
|
||||||
Height="{StaticResource MenuItemHeight}"
|
x:Name="menuMoveBottom"
|
||||||
Header="{x:Static resx:ResUI.menuMoveBottom}" />
|
Height="{StaticResource MenuItemHeight}"
|
||||||
|
Header="{x:Static resx:ResUI.menuMoveBottom}" />
|
||||||
|
</MenuItem>
|
||||||
<MenuItem
|
<MenuItem
|
||||||
x:Name="menuSelectAll"
|
x:Name="menuSelectAll"
|
||||||
Height="{StaticResource MenuItemHeight}"
|
Height="{StaticResource MenuItemHeight}"
|
||||||
@@ -588,7 +617,7 @@
|
|||||||
</Style>
|
</Style>
|
||||||
</DataGrid.Resources>
|
</DataGrid.Resources>
|
||||||
<DataGrid.Columns>
|
<DataGrid.Columns>
|
||||||
<DataGridTemplateColumn Width="50">
|
<DataGridTemplateColumn Width="40">
|
||||||
<DataGridTemplateColumn.CellTemplate>
|
<DataGridTemplateColumn.CellTemplate>
|
||||||
<DataTemplate>
|
<DataTemplate>
|
||||||
<Border
|
<Border
|
||||||
@@ -600,41 +629,50 @@
|
|||||||
</DataGridTemplateColumn.CellTemplate>
|
</DataGridTemplateColumn.CellTemplate>
|
||||||
</DataGridTemplateColumn>
|
</DataGridTemplateColumn>
|
||||||
|
|
||||||
<materialDesign:DataGridTextColumn
|
<base:MyDGTextColumn
|
||||||
Width="80"
|
Width="80"
|
||||||
Binding="{Binding configType}"
|
Binding="{Binding configType}"
|
||||||
|
ExName="configType"
|
||||||
Header="{x:Static resx:ResUI.LvServiceType}" />
|
Header="{x:Static resx:ResUI.LvServiceType}" />
|
||||||
<DataGridTextColumn
|
<base:MyDGTextColumn
|
||||||
Width="150"
|
Width="150"
|
||||||
Binding="{Binding remarks}"
|
Binding="{Binding remarks}"
|
||||||
|
ExName="remarks"
|
||||||
Header="{x:Static resx:ResUI.LvAlias}" />
|
Header="{x:Static resx:ResUI.LvAlias}" />
|
||||||
<DataGridTextColumn
|
<base:MyDGTextColumn
|
||||||
Width="120"
|
Width="120"
|
||||||
Binding="{Binding address}"
|
Binding="{Binding address}"
|
||||||
|
ExName="address"
|
||||||
Header="{x:Static resx:ResUI.LvAddress}" />
|
Header="{x:Static resx:ResUI.LvAddress}" />
|
||||||
<DataGridTextColumn
|
<base:MyDGTextColumn
|
||||||
Width="60"
|
Width="60"
|
||||||
Binding="{Binding port}"
|
Binding="{Binding port}"
|
||||||
|
ExName="port"
|
||||||
Header="{x:Static resx:ResUI.LvPort}" />
|
Header="{x:Static resx:ResUI.LvPort}" />
|
||||||
<DataGridTextColumn
|
<base:MyDGTextColumn
|
||||||
Width="100"
|
Width="100"
|
||||||
Binding="{Binding security}"
|
Binding="{Binding security}"
|
||||||
|
ExName="security"
|
||||||
Header="{x:Static resx:ResUI.LvEncryptionMethod}" />
|
Header="{x:Static resx:ResUI.LvEncryptionMethod}" />
|
||||||
<DataGridTextColumn
|
<base:MyDGTextColumn
|
||||||
Width="100"
|
Width="100"
|
||||||
Binding="{Binding network}"
|
Binding="{Binding network}"
|
||||||
|
ExName="network"
|
||||||
Header="{x:Static resx:ResUI.LvTransportProtocol}" />
|
Header="{x:Static resx:ResUI.LvTransportProtocol}" />
|
||||||
<DataGridTextColumn
|
<base:MyDGTextColumn
|
||||||
Width="100"
|
Width="100"
|
||||||
Binding="{Binding streamSecurity}"
|
Binding="{Binding streamSecurity}"
|
||||||
|
ExName="streamSecurity"
|
||||||
Header="{x:Static resx:ResUI.LvTLS}" />
|
Header="{x:Static resx:ResUI.LvTLS}" />
|
||||||
<DataGridTextColumn
|
<base:MyDGTextColumn
|
||||||
Width="100"
|
Width="100"
|
||||||
Binding="{Binding subRemarks}"
|
Binding="{Binding subRemarks}"
|
||||||
|
ExName="subRemarks"
|
||||||
Header="{x:Static resx:ResUI.LvSubscription}" />
|
Header="{x:Static resx:ResUI.LvSubscription}" />
|
||||||
<DataGridTextColumn
|
<base:MyDGTextColumn
|
||||||
Width="100"
|
Width="100"
|
||||||
Binding="{Binding delayVal}"
|
Binding="{Binding delayVal}"
|
||||||
|
ExName="delayVal"
|
||||||
Header="{x:Static resx:ResUI.LvTestDelay}">
|
Header="{x:Static resx:ResUI.LvTestDelay}">
|
||||||
<DataGridTextColumn.ElementStyle>
|
<DataGridTextColumn.ElementStyle>
|
||||||
<Style TargetType="{x:Type TextBlock}">
|
<Style TargetType="{x:Type TextBlock}">
|
||||||
@@ -642,37 +680,42 @@
|
|||||||
<Setter Property="Foreground" Value="{Binding delay, Converter={StaticResource DelayColorConverter}}" />
|
<Setter Property="Foreground" Value="{Binding delay, Converter={StaticResource DelayColorConverter}}" />
|
||||||
</Style>
|
</Style>
|
||||||
</DataGridTextColumn.ElementStyle>
|
</DataGridTextColumn.ElementStyle>
|
||||||
</DataGridTextColumn>
|
</base:MyDGTextColumn>
|
||||||
<DataGridTextColumn
|
<base:MyDGTextColumn
|
||||||
Width="100"
|
Width="100"
|
||||||
Binding="{Binding speedVal}"
|
Binding="{Binding speedVal}"
|
||||||
|
ExName="speedVal"
|
||||||
Header="{x:Static resx:ResUI.LvTestSpeed}">
|
Header="{x:Static resx:ResUI.LvTestSpeed}">
|
||||||
<DataGridTextColumn.ElementStyle>
|
<DataGridTextColumn.ElementStyle>
|
||||||
<Style TargetType="{x:Type TextBlock}">
|
<Style TargetType="{x:Type TextBlock}">
|
||||||
<Setter Property="HorizontalAlignment" Value="Right" />
|
<Setter Property="HorizontalAlignment" Value="Right" />
|
||||||
</Style>
|
</Style>
|
||||||
</DataGridTextColumn.ElementStyle>
|
</DataGridTextColumn.ElementStyle>
|
||||||
</DataGridTextColumn>
|
</base:MyDGTextColumn>
|
||||||
|
|
||||||
<DataGridTextColumn
|
<base:MyDGTextColumn
|
||||||
x:Name="colTodayUp"
|
x:Name="colTodayUp"
|
||||||
Width="100"
|
Width="100"
|
||||||
Binding="{Binding todayUp}"
|
Binding="{Binding todayUp}"
|
||||||
|
ExName="todayUp"
|
||||||
Header="{x:Static resx:ResUI.LvTodayUploadDataAmount}" />
|
Header="{x:Static resx:ResUI.LvTodayUploadDataAmount}" />
|
||||||
<DataGridTextColumn
|
<base:MyDGTextColumn
|
||||||
x:Name="colTodayDown"
|
x:Name="colTodayDown"
|
||||||
Width="100"
|
Width="100"
|
||||||
Binding="{Binding todayDown}"
|
Binding="{Binding todayDown}"
|
||||||
|
ExName="todayDown"
|
||||||
Header="{x:Static resx:ResUI.LvTodayDownloadDataAmount}" />
|
Header="{x:Static resx:ResUI.LvTodayDownloadDataAmount}" />
|
||||||
<DataGridTextColumn
|
<base:MyDGTextColumn
|
||||||
x:Name="colTotalUp"
|
x:Name="colTotalUp"
|
||||||
Width="100"
|
Width="100"
|
||||||
Binding="{Binding totalUp}"
|
Binding="{Binding totalUp}"
|
||||||
|
ExName="totalUp"
|
||||||
Header="{x:Static resx:ResUI.LvTotalUploadDataAmount}" />
|
Header="{x:Static resx:ResUI.LvTotalUploadDataAmount}" />
|
||||||
<DataGridTextColumn
|
<base:MyDGTextColumn
|
||||||
x:Name="colTotalDown"
|
x:Name="colTotalDown"
|
||||||
Width="100"
|
Width="100"
|
||||||
Binding="{Binding totalDown}"
|
Binding="{Binding totalDown}"
|
||||||
|
ExName="totalDown"
|
||||||
Header="{x:Static resx:ResUI.LvTotalDownloadDataAmount}" />
|
Header="{x:Static resx:ResUI.LvTotalDownloadDataAmount}" />
|
||||||
|
|
||||||
</DataGrid.Columns>
|
</DataGrid.Columns>
|
||||||
@@ -694,6 +737,7 @@
|
|||||||
<materialDesign:PackIcon
|
<materialDesign:PackIcon
|
||||||
x:Name="menuSystemProxyClear2"
|
x:Name="menuSystemProxyClear2"
|
||||||
Margin="0,0,8,0"
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
Kind="Check" />
|
Kind="Check" />
|
||||||
<TextBlock Text="{x:Static resx:ResUI.menuSystemProxyClear}" />
|
<TextBlock Text="{x:Static resx:ResUI.menuSystemProxyClear}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
@@ -705,6 +749,7 @@
|
|||||||
<materialDesign:PackIcon
|
<materialDesign:PackIcon
|
||||||
x:Name="menuSystemProxySet2"
|
x:Name="menuSystemProxySet2"
|
||||||
Margin="0,0,8,0"
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
Kind="Check" />
|
Kind="Check" />
|
||||||
<TextBlock Text="{x:Static resx:ResUI.menuSystemProxySet}" />
|
<TextBlock Text="{x:Static resx:ResUI.menuSystemProxySet}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
@@ -716,6 +761,7 @@
|
|||||||
<materialDesign:PackIcon
|
<materialDesign:PackIcon
|
||||||
x:Name="menuSystemProxyNothing2"
|
x:Name="menuSystemProxyNothing2"
|
||||||
Margin="0,0,8,0"
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
Kind="Check" />
|
Kind="Check" />
|
||||||
<TextBlock Text="{x:Static resx:ResUI.menuSystemProxyNothing}" />
|
<TextBlock Text="{x:Static resx:ResUI.menuSystemProxyNothing}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
@@ -727,6 +773,7 @@
|
|||||||
<materialDesign:PackIcon
|
<materialDesign:PackIcon
|
||||||
x:Name="menuSystemProxyPac2"
|
x:Name="menuSystemProxyPac2"
|
||||||
Margin="0,0,8,0"
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
Kind="Check" />
|
Kind="Check" />
|
||||||
<TextBlock Text="{x:Static resx:ResUI.menuSystemProxyPac}" />
|
<TextBlock Text="{x:Static resx:ResUI.menuSystemProxyPac}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
@@ -784,6 +831,22 @@
|
|||||||
Header="{x:Static resx:ResUI.menuExit}" />
|
Header="{x:Static resx:ResUI.menuExit}" />
|
||||||
</ContextMenu>
|
</ContextMenu>
|
||||||
</tb:TaskbarIcon.ContextMenu>
|
</tb:TaskbarIcon.ContextMenu>
|
||||||
|
<tb:TaskbarIcon.TrayToolTip>
|
||||||
|
<Border
|
||||||
|
Width="Auto"
|
||||||
|
Height="Auto"
|
||||||
|
Background="{DynamicResource MaterialDesignLightBackground}"
|
||||||
|
BorderBrush="{DynamicResource MaterialDesignDarkBackground}"
|
||||||
|
BorderThickness="0"
|
||||||
|
CornerRadius="4">
|
||||||
|
<TextBlock
|
||||||
|
Margin="8"
|
||||||
|
HorizontalAlignment="Center"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Foreground="{DynamicResource MaterialDesignDarkBackground}"
|
||||||
|
Text="{Binding Mode=OneWay, Path=ToolTipText}" />
|
||||||
|
</Border>
|
||||||
|
</tb:TaskbarIcon.TrayToolTip>
|
||||||
</tb:TaskbarIcon>
|
</tb:TaskbarIcon>
|
||||||
<materialDesign:Snackbar x:Name="MainSnackbar" MessageQueue="{materialDesign:MessageQueue}" />
|
<materialDesign:Snackbar x:Name="MainSnackbar" MessageQueue="{materialDesign:MessageQueue}" />
|
||||||
</Grid>
|
</Grid>
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ using System.Windows.Controls.Primitives;
|
|||||||
using System.Windows.Input;
|
using System.Windows.Input;
|
||||||
using System.Windows.Interop;
|
using System.Windows.Interop;
|
||||||
using System.Windows.Media;
|
using System.Windows.Media;
|
||||||
|
using v2rayN.Base;
|
||||||
using v2rayN.Handler;
|
using v2rayN.Handler;
|
||||||
using v2rayN.Mode;
|
using v2rayN.Mode;
|
||||||
using v2rayN.Resx;
|
using v2rayN.Resx;
|
||||||
@@ -149,6 +150,7 @@ namespace v2rayN.Views
|
|||||||
|
|
||||||
this.OneWayBind(ViewModel, vm => vm.Servers, v => v.cmbServers.ItemsSource).DisposeWith(disposables);
|
this.OneWayBind(ViewModel, vm => vm.Servers, v => v.cmbServers.ItemsSource).DisposeWith(disposables);
|
||||||
this.Bind(ViewModel, vm => vm.SelectedServer, v => v.cmbServers.SelectedItem).DisposeWith(disposables);
|
this.Bind(ViewModel, vm => vm.SelectedServer, v => v.cmbServers.SelectedItem).DisposeWith(disposables);
|
||||||
|
this.OneWayBind(ViewModel, vm => vm.BlServers, v => v.cmbServers.Visibility).DisposeWith(disposables);
|
||||||
|
|
||||||
//tray menu
|
//tray menu
|
||||||
this.BindCommand(ViewModel, vm => vm.AddServerViaClipboardCmd, v => v.menuAddServerViaClipboard2).DisposeWith(disposables);
|
this.BindCommand(ViewModel, vm => vm.AddServerViaClipboardCmd, v => v.menuAddServerViaClipboard2).DisposeWith(disposables);
|
||||||
@@ -157,6 +159,7 @@ namespace v2rayN.Views
|
|||||||
this.BindCommand(ViewModel, vm => vm.SubUpdateViaProxyCmd, v => v.menuSubUpdateViaProxy2).DisposeWith(disposables);
|
this.BindCommand(ViewModel, vm => vm.SubUpdateViaProxyCmd, v => v.menuSubUpdateViaProxy2).DisposeWith(disposables);
|
||||||
|
|
||||||
this.OneWayBind(ViewModel, vm => vm.NotifyIcon, v => v.tbNotify.Icon).DisposeWith(disposables);
|
this.OneWayBind(ViewModel, vm => vm.NotifyIcon, v => v.tbNotify.Icon).DisposeWith(disposables);
|
||||||
|
this.OneWayBind(ViewModel, vm => vm.RunningServerToolTipText, v => v.tbNotify.ToolTipText).DisposeWith(disposables);
|
||||||
this.OneWayBind(ViewModel, vm => vm.NotifyLeftClickCmd, v => v.tbNotify.LeftClickCommand).DisposeWith(disposables);
|
this.OneWayBind(ViewModel, vm => vm.NotifyLeftClickCmd, v => v.tbNotify.LeftClickCommand).DisposeWith(disposables);
|
||||||
this.OneWayBind(ViewModel, vm => vm.AppIcon, v => v.Icon).DisposeWith(disposables);
|
this.OneWayBind(ViewModel, vm => vm.AppIcon, v => v.Icon).DisposeWith(disposables);
|
||||||
|
|
||||||
@@ -189,6 +192,16 @@ namespace v2rayN.Views
|
|||||||
this.Title = $"{Utils.GetVersion()} - {(IsAdministrator ? ResUI.RunAsAdmin : ResUI.NotRunAsAdmin)}";
|
this.Title = $"{Utils.GetVersion()} - {(IsAdministrator ? ResUI.RunAsAdmin : ResUI.NotRunAsAdmin)}";
|
||||||
|
|
||||||
spEnableTun.Visibility = IsAdministrator ? Visibility.Visible : Visibility.Collapsed;
|
spEnableTun.Visibility = IsAdministrator ? Visibility.Visible : Visibility.Collapsed;
|
||||||
|
|
||||||
|
if (_config.uiItem.autoHideStartup)
|
||||||
|
{
|
||||||
|
WindowState = WindowState.Minimized;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!_config.guiItem.enableHWA)
|
||||||
|
{
|
||||||
|
RenderOptions.ProcessRenderMode = RenderMode.SoftwareOnly;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#region Event
|
#region Event
|
||||||
@@ -197,13 +210,13 @@ namespace v2rayN.Views
|
|||||||
{
|
{
|
||||||
if (action == "AdjustMainLvColWidth")
|
if (action == "AdjustMainLvColWidth")
|
||||||
{
|
{
|
||||||
Application.Current.Dispatcher.Invoke((Action)(() =>
|
Application.Current.Dispatcher.Invoke(() =>
|
||||||
{
|
{
|
||||||
foreach (var it in lstProfiles.Columns)
|
foreach (var it in lstProfiles.Columns)
|
||||||
{
|
{
|
||||||
it.Width = new DataGridLength(1, DataGridLengthUnitType.Auto);
|
it.Width = new DataGridLength(1, DataGridLengthUnitType.Auto);
|
||||||
}
|
}
|
||||||
}));
|
});
|
||||||
}
|
}
|
||||||
else if (action == "ProfilesFocus")
|
else if (action == "ProfilesFocus")
|
||||||
{
|
{
|
||||||
@@ -265,14 +278,7 @@ namespace v2rayN.Views
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
//find index
|
if (colHeader.Column.GetType().Name != typeof(MyDGTextColumn).Name)
|
||||||
var index = lstProfiles.Columns.IndexOf(colHeader.Column);
|
|
||||||
if (index < 0)
|
|
||||||
{
|
|
||||||
index = colHeader.TabIndex;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (index == 0)
|
|
||||||
{
|
{
|
||||||
foreach (var it in lstProfiles.Columns)
|
foreach (var it in lstProfiles.Columns)
|
||||||
{
|
{
|
||||||
@@ -282,7 +288,8 @@ namespace v2rayN.Views
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ViewModel?.SortServer(index);
|
var colName = ((MyDGTextColumn)colHeader.Column).ExName;
|
||||||
|
ViewModel?.SortServer(colName);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void menuSelectAll_Click(object sender, RoutedEventArgs e)
|
private void menuSelectAll_Click(object sender, RoutedEventArgs e)
|
||||||
@@ -355,7 +362,7 @@ namespace v2rayN.Views
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (e.Key == Key.Enter || e.Key == Key.Return)
|
if (e.Key is Key.Enter or Key.Return)
|
||||||
{
|
{
|
||||||
ViewModel?.SetDefaultServer();
|
ViewModel?.SetDefaultServer();
|
||||||
}
|
}
|
||||||
@@ -403,7 +410,7 @@ namespace v2rayN.Views
|
|||||||
}
|
}
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region UI
|
#region UI
|
||||||
|
|
||||||
private void RestoreUI()
|
private void RestoreUI()
|
||||||
{
|
{
|
||||||
@@ -429,11 +436,28 @@ namespace v2rayN.Views
|
|||||||
gridMain.RowDefinitions[2].Height = new GridLength(_config.uiItem.mainGirdHeight2, GridUnitType.Star);
|
gridMain.RowDefinitions[2].Height = new GridLength(_config.uiItem.mainGirdHeight2, GridUnitType.Star);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int k = 0; k < lstProfiles.Columns.Count; k++)
|
var lvColumnItem = _config.uiItem.mainColumnItem.OrderBy(t => t.Index).ToList();
|
||||||
|
for (int i = 0; i < lvColumnItem.Count; i++)
|
||||||
{
|
{
|
||||||
var width = ConfigHandler.GetformMainLvColWidth(ref _config, ((EServerColName)k).ToString(), Convert.ToInt32(lstProfiles.Columns[k].Width.Value));
|
var item = lvColumnItem[i];
|
||||||
lstProfiles.Columns[k].Width = width;
|
for (int k = 1; k < lstProfiles.Columns.Count; k++)
|
||||||
|
{
|
||||||
|
var item2 = (MyDGTextColumn)lstProfiles.Columns[k];
|
||||||
|
if (item2.ExName == item.Name)
|
||||||
|
{
|
||||||
|
if (item.Width <= 0)
|
||||||
|
{
|
||||||
|
item2.Visibility = Visibility.Hidden;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
item2.Width = item.Width;
|
||||||
|
item2.DisplayIndex = i + 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!_config.guiItem.enableStatistics)
|
if (!_config.guiItem.enableStatistics)
|
||||||
{
|
{
|
||||||
colTodayUp.Visibility = Visibility.Hidden;
|
colTodayUp.Visibility = Visibility.Hidden;
|
||||||
@@ -447,10 +471,19 @@ namespace v2rayN.Views
|
|||||||
_config.uiItem.mainWidth = this.Width;
|
_config.uiItem.mainWidth = this.Width;
|
||||||
_config.uiItem.mainHeight = this.Height;
|
_config.uiItem.mainHeight = this.Height;
|
||||||
|
|
||||||
for (int k = 0; k < lstProfiles.Columns.Count; k++)
|
List<ColumnItem> lvColumnItem = new();
|
||||||
|
for (int k = 1; k < lstProfiles.Columns.Count; k++)
|
||||||
{
|
{
|
||||||
ConfigHandler.AddformMainLvColWidth(ref _config, ((EServerColName)k).ToString(), Convert.ToInt32(lstProfiles.Columns[k].ActualWidth));
|
var item2 = (MyDGTextColumn)lstProfiles.Columns[k];
|
||||||
|
lvColumnItem.Add(new()
|
||||||
|
{
|
||||||
|
Name = item2.ExName,
|
||||||
|
Width = item2.Visibility == Visibility.Visible ? Convert.ToInt32(item2.ActualWidth) : 0,
|
||||||
|
Index = item2.DisplayIndex
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
_config.uiItem.mainColumnItem = lvColumnItem;
|
||||||
|
|
||||||
_config.uiItem.mainGirdHeight1 = Math.Ceiling(gridMain.RowDefinitions[0].ActualHeight + 0.1);
|
_config.uiItem.mainGirdHeight1 = Math.Ceiling(gridMain.RowDefinitions[0].ActualHeight + 0.1);
|
||||||
_config.uiItem.mainGirdHeight2 = Math.Ceiling(gridMain.RowDefinitions[2].ActualHeight + 0.1);
|
_config.uiItem.mainGirdHeight2 = Math.Ceiling(gridMain.RowDefinitions[2].ActualHeight + 0.1);
|
||||||
}
|
}
|
||||||
@@ -471,9 +504,8 @@ namespace v2rayN.Views
|
|||||||
}
|
}
|
||||||
private void MenuItem_Click(object sender, RoutedEventArgs e)
|
private void MenuItem_Click(object sender, RoutedEventArgs e)
|
||||||
{
|
{
|
||||||
if (sender is MenuItem)
|
if (sender is MenuItem item)
|
||||||
{
|
{
|
||||||
MenuItem item = (MenuItem)sender;
|
|
||||||
Utils.ProcessStart(item.Tag.ToString());
|
Utils.ProcessStart(item.Tag.ToString());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -482,7 +514,7 @@ namespace v2rayN.Views
|
|||||||
#endregion
|
#endregion
|
||||||
#region Drag and Drop
|
#region Drag and Drop
|
||||||
|
|
||||||
private Point startPoint = new Point();
|
private Point startPoint = new();
|
||||||
private int startIndex = -1;
|
private int startIndex = -1;
|
||||||
private string formatData = "ProfileItemModel";
|
private string formatData = "ProfileItemModel";
|
||||||
|
|
||||||
@@ -523,8 +555,7 @@ namespace v2rayN.Views
|
|||||||
Math.Abs(diff.Y) > SystemParameters.MinimumVerticalDragDistance))
|
Math.Abs(diff.Y) > SystemParameters.MinimumVerticalDragDistance))
|
||||||
{
|
{
|
||||||
// Get the dragged Item
|
// Get the dragged Item
|
||||||
var listView = sender as DataGrid;
|
if (sender is not DataGrid listView) return;
|
||||||
if (listView == null) return;
|
|
||||||
var listViewItem = FindAnchestor<DataGridRow>((DependencyObject)e.OriginalSource);
|
var listViewItem = FindAnchestor<DataGridRow>((DependencyObject)e.OriginalSource);
|
||||||
if (listViewItem == null) return; // Abort
|
if (listViewItem == null) return; // Abort
|
||||||
// Find the data behind the ListViewItem
|
// Find the data behind the ListViewItem
|
||||||
@@ -532,7 +563,7 @@ namespace v2rayN.Views
|
|||||||
if (item == null) return; // Abort
|
if (item == null) return; // Abort
|
||||||
// Initialize the drag & drop operation
|
// Initialize the drag & drop operation
|
||||||
startIndex = lstProfiles.SelectedIndex;
|
startIndex = lstProfiles.SelectedIndex;
|
||||||
DataObject dragData = new DataObject(formatData, item);
|
DataObject dragData = new(formatData, item);
|
||||||
DragDrop.DoDragDrop(listViewItem, dragData, DragDropEffects.Copy | DragDropEffects.Move);
|
DragDrop.DoDragDrop(listViewItem, dragData, DragDropEffects.Copy | DragDropEffects.Move);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -550,8 +581,7 @@ namespace v2rayN.Views
|
|||||||
if (e.Data.GetDataPresent(formatData) && sender == e.Source)
|
if (e.Data.GetDataPresent(formatData) && sender == e.Source)
|
||||||
{
|
{
|
||||||
// Get the drop Item destination
|
// Get the drop Item destination
|
||||||
var listView = sender as DataGrid;
|
if (sender is not DataGrid listView) return;
|
||||||
if (listView == null) return;
|
|
||||||
var listViewItem = FindAnchestor<DataGridRow>((DependencyObject)e.OriginalSource);
|
var listViewItem = FindAnchestor<DataGridRow>((DependencyObject)e.OriginalSource);
|
||||||
if (listViewItem == null)
|
if (listViewItem == null)
|
||||||
{
|
{
|
||||||
@@ -576,4 +606,4 @@ namespace v2rayN.Views
|
|||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -48,6 +48,7 @@
|
|||||||
FontSize="{DynamicResource StdFontSizeMsg}"
|
FontSize="{DynamicResource StdFontSizeMsg}"
|
||||||
HorizontalScrollBarVisibility="Auto"
|
HorizontalScrollBarVisibility="Auto"
|
||||||
IsReadOnly="True"
|
IsReadOnly="True"
|
||||||
|
IsReadOnlyCaretVisible="True"
|
||||||
TextAlignment="Left"
|
TextAlignment="Left"
|
||||||
TextWrapping="Wrap"
|
TextWrapping="Wrap"
|
||||||
VerticalScrollBarVisibility="Visible">
|
VerticalScrollBarVisibility="Visible">
|
||||||
|
|||||||
@@ -11,6 +11,10 @@ namespace v2rayN.Views
|
|||||||
public partial class MsgView
|
public partial class MsgView
|
||||||
{
|
{
|
||||||
private static Config _config;
|
private static Config _config;
|
||||||
|
|
||||||
|
private string lastMsgFilter;
|
||||||
|
private bool lastMsgFilterNotAvailable;
|
||||||
|
|
||||||
public MsgView()
|
public MsgView()
|
||||||
{
|
{
|
||||||
InitializeComponent();
|
InitializeComponent();
|
||||||
@@ -28,28 +32,38 @@ namespace v2rayN.Views
|
|||||||
|
|
||||||
void DelegateAppendText(string msg)
|
void DelegateAppendText(string msg)
|
||||||
{
|
{
|
||||||
Dispatcher.BeginInvoke(new Action<string>(AppendText), DispatcherPriority.Send, msg);
|
Dispatcher.BeginInvoke(AppendText, DispatcherPriority.Send, msg);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void AppendText(string msg)
|
public void AppendText(string msg)
|
||||||
{
|
{
|
||||||
if (msg.Equals(Global.CommandClearMsg))
|
if (msg == Global.CommandClearMsg)
|
||||||
{
|
{
|
||||||
ClearMsg();
|
ClearMsg();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (!togAutoRefresh.IsChecked.Value)
|
if (togAutoRefresh.IsChecked == false)
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var MsgFilter = cmbMsgFilter.Text.TrimEx();
|
var MsgFilter = cmbMsgFilter.Text.TrimEx();
|
||||||
if (!Utils.IsNullOrEmpty(MsgFilter))
|
if (MsgFilter != lastMsgFilter) lastMsgFilterNotAvailable = false;
|
||||||
|
if (!string.IsNullOrEmpty(MsgFilter) && !lastMsgFilterNotAvailable)
|
||||||
{
|
{
|
||||||
if (!Regex.IsMatch(msg, MsgFilter))
|
try
|
||||||
{
|
{
|
||||||
return;
|
if (!Regex.IsMatch(msg, MsgFilter)) // <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ʽ<EFBFBD><CABD><EFBFBD>쳣
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception)
|
||||||
|
{
|
||||||
|
lastMsgFilterNotAvailable = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
lastMsgFilter = MsgFilter;
|
||||||
|
|
||||||
ShowMsg(msg);
|
ShowMsg(msg);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,7 +10,7 @@
|
|||||||
xmlns:resx="clr-namespace:v2rayN.Resx"
|
xmlns:resx="clr-namespace:v2rayN.Resx"
|
||||||
xmlns:vms="clr-namespace:v2rayN.ViewModels"
|
xmlns:vms="clr-namespace:v2rayN.ViewModels"
|
||||||
Title="{x:Static resx:ResUI.menuSetting}"
|
Title="{x:Static resx:ResUI.menuSetting}"
|
||||||
Width="900"
|
Width="1000"
|
||||||
Height="700"
|
Height="700"
|
||||||
x:TypeArguments="vms:OptionSettingViewModel"
|
x:TypeArguments="vms:OptionSettingViewModel"
|
||||||
Background="{DynamicResource MaterialDesignPaper}"
|
Background="{DynamicResource MaterialDesignPaper}"
|
||||||
@@ -23,7 +23,7 @@
|
|||||||
TextOptions.TextRenderingMode="Auto"
|
TextOptions.TextRenderingMode="Auto"
|
||||||
WindowStartupLocation="CenterScreen"
|
WindowStartupLocation="CenterScreen"
|
||||||
mc:Ignorable="d">
|
mc:Ignorable="d">
|
||||||
<DockPanel Margin="16">
|
<DockPanel Margin="8">
|
||||||
<Grid HorizontalAlignment="Center" DockPanel.Dock="Bottom">
|
<Grid HorizontalAlignment="Center" DockPanel.Dock="Bottom">
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
<ColumnDefinition Width="200" />
|
<ColumnDefinition Width="200" />
|
||||||
@@ -47,7 +47,7 @@
|
|||||||
Style="{StaticResource DefButton}" />
|
Style="{StaticResource DefButton}" />
|
||||||
</Grid>
|
</Grid>
|
||||||
|
|
||||||
<TabControl>
|
<TabControl HorizontalContentAlignment="Left">
|
||||||
<TabItem Header="{x:Static resx:ResUI.TbSettingsCore}">
|
<TabItem Header="{x:Static resx:ResUI.TbSettingsCore}">
|
||||||
<ScrollViewer VerticalScrollBarVisibility="Visible">
|
<ScrollViewer VerticalScrollBarVisibility="Visible">
|
||||||
<Grid Margin="{StaticResource SettingItemMargin}">
|
<Grid Margin="{StaticResource SettingItemMargin}">
|
||||||
@@ -481,11 +481,12 @@
|
|||||||
<RowDefinition Height="Auto" />
|
<RowDefinition Height="Auto" />
|
||||||
<RowDefinition Height="Auto" />
|
<RowDefinition Height="Auto" />
|
||||||
<RowDefinition Height="Auto" />
|
<RowDefinition Height="Auto" />
|
||||||
|
<RowDefinition Height="Auto" />
|
||||||
</Grid.RowDefinitions>
|
</Grid.RowDefinitions>
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
<ColumnDefinition Width="Auto" />
|
<ColumnDefinition Width="Auto" />
|
||||||
<ColumnDefinition Width="Auto" />
|
<ColumnDefinition Width="Auto" />
|
||||||
<ColumnDefinition Width="Auto" />
|
<ColumnDefinition Width="*" />
|
||||||
</Grid.ColumnDefinitions>
|
</Grid.ColumnDefinitions>
|
||||||
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
@@ -507,7 +508,8 @@
|
|||||||
Margin="{StaticResource ServerItemMargin}"
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Style="{StaticResource ToolbarTextBlock}"
|
Style="{StaticResource ToolbarTextBlock}"
|
||||||
Text="{x:Static resx:ResUI.TbSettingsStartBootTip}" />
|
Text="{x:Static resx:ResUI.TbSettingsStartBootTip}"
|
||||||
|
TextWrapping="Wrap" />
|
||||||
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Grid.Row="2"
|
Grid.Row="2"
|
||||||
@@ -719,7 +721,8 @@
|
|||||||
Margin="{StaticResource ServerItemMargin}"
|
Margin="{StaticResource ServerItemMargin}"
|
||||||
VerticalAlignment="Center"
|
VerticalAlignment="Center"
|
||||||
Style="{StaticResource ToolbarTextBlock}"
|
Style="{StaticResource ToolbarTextBlock}"
|
||||||
Text="{x:Static resx:ResUI.TbSettingsCurrentFontFamilyTip}" />
|
Text="{x:Static resx:ResUI.TbSettingsCurrentFontFamilyTip}"
|
||||||
|
TextWrapping="Wrap" />
|
||||||
|
|
||||||
<TextBlock
|
<TextBlock
|
||||||
Grid.Row="16"
|
Grid.Row="16"
|
||||||
@@ -750,6 +753,20 @@
|
|||||||
Width="300"
|
Width="300"
|
||||||
Margin="{StaticResource SettingItemMargin}"
|
Margin="{StaticResource SettingItemMargin}"
|
||||||
Style="{StaticResource DefComboBox}" />
|
Style="{StaticResource DefComboBox}" />
|
||||||
|
|
||||||
|
<TextBlock
|
||||||
|
Grid.Row="18"
|
||||||
|
Grid.Column="0"
|
||||||
|
Margin="{StaticResource SettingItemMargin}"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Style="{StaticResource ToolbarTextBlock}"
|
||||||
|
Text="{x:Static resx:ResUI.TbSettingsEnableHWA}" />
|
||||||
|
<ToggleButton
|
||||||
|
x:Name="togEnableHWA"
|
||||||
|
Grid.Row="18"
|
||||||
|
Grid.Column="1"
|
||||||
|
Margin="{StaticResource SettingItemMargin}"
|
||||||
|
HorizontalAlignment="Left" />
|
||||||
</Grid>
|
</Grid>
|
||||||
</ScrollViewer>
|
</ScrollViewer>
|
||||||
</TabItem>
|
</TabItem>
|
||||||
@@ -941,13 +958,15 @@
|
|||||||
|
|
||||||
<Grid
|
<Grid
|
||||||
x:Name="gridTunModeDirect"
|
x:Name="gridTunModeDirect"
|
||||||
Width="600"
|
Width="800"
|
||||||
Margin="{StaticResource SettingItemMargin}"
|
Margin="{StaticResource SettingItemMargin}"
|
||||||
HorizontalAlignment="Left">
|
HorizontalAlignment="Left">
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
<ColumnDefinition Width="1*" />
|
<ColumnDefinition Width="2*" />
|
||||||
<ColumnDefinition Width="10" />
|
<ColumnDefinition Width="5" />
|
||||||
<ColumnDefinition Width="1*" />
|
<ColumnDefinition Width="2*" />
|
||||||
|
<ColumnDefinition Width="5" />
|
||||||
|
<ColumnDefinition Width="3*" />
|
||||||
</Grid.ColumnDefinitions>
|
</Grid.ColumnDefinitions>
|
||||||
<GroupBox
|
<GroupBox
|
||||||
Grid.Column="0"
|
Grid.Column="0"
|
||||||
@@ -972,17 +991,31 @@
|
|||||||
TextWrapping="Wrap"
|
TextWrapping="Wrap"
|
||||||
VerticalScrollBarVisibility="Auto" />
|
VerticalScrollBarVisibility="Auto" />
|
||||||
</GroupBox>
|
</GroupBox>
|
||||||
|
<GridSplitter Grid.Column="3" HorizontalAlignment="Stretch" />
|
||||||
|
<GroupBox
|
||||||
|
Grid.Column="4"
|
||||||
|
Header="{x:Static resx:ResUI.TbSettingsTunModeDNS}"
|
||||||
|
Style="{StaticResource MyGroupBox}">
|
||||||
|
<TextBox
|
||||||
|
Name="txtDirectDNS"
|
||||||
|
AcceptsReturn="True"
|
||||||
|
Style="{StaticResource DefTextBox}"
|
||||||
|
TextWrapping="Wrap"
|
||||||
|
VerticalScrollBarVisibility="Auto" />
|
||||||
|
</GroupBox>
|
||||||
</Grid>
|
</Grid>
|
||||||
|
|
||||||
<Grid
|
<Grid
|
||||||
x:Name="gridTunModeProxy"
|
x:Name="gridTunModeProxy"
|
||||||
Width="600"
|
Width="800"
|
||||||
Margin="{StaticResource SettingItemMargin}"
|
Margin="{StaticResource SettingItemMargin}"
|
||||||
HorizontalAlignment="Left">
|
HorizontalAlignment="Left">
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
<ColumnDefinition Width="1*" />
|
<ColumnDefinition Width="2*" />
|
||||||
<ColumnDefinition Width="10" />
|
<ColumnDefinition Width="5" />
|
||||||
<ColumnDefinition Width="1*" />
|
<ColumnDefinition Width="2*" />
|
||||||
|
<ColumnDefinition Width="5" />
|
||||||
|
<ColumnDefinition Width="3*" />
|
||||||
</Grid.ColumnDefinitions>
|
</Grid.ColumnDefinitions>
|
||||||
<GroupBox
|
<GroupBox
|
||||||
Grid.Column="0"
|
Grid.Column="0"
|
||||||
@@ -1007,6 +1040,18 @@
|
|||||||
TextWrapping="Wrap"
|
TextWrapping="Wrap"
|
||||||
VerticalScrollBarVisibility="Auto" />
|
VerticalScrollBarVisibility="Auto" />
|
||||||
</GroupBox>
|
</GroupBox>
|
||||||
|
<GridSplitter Grid.Column="3" HorizontalAlignment="Stretch" />
|
||||||
|
<GroupBox
|
||||||
|
Grid.Column="4"
|
||||||
|
Header="{x:Static resx:ResUI.TbSettingsTunModeDNS}"
|
||||||
|
Style="{StaticResource MyGroupBox}">
|
||||||
|
<TextBox
|
||||||
|
Name="txtProxyDNS"
|
||||||
|
AcceptsReturn="True"
|
||||||
|
Style="{StaticResource DefTextBox}"
|
||||||
|
TextWrapping="Wrap"
|
||||||
|
VerticalScrollBarVisibility="Auto" />
|
||||||
|
</GroupBox>
|
||||||
</Grid>
|
</Grid>
|
||||||
|
|
||||||
</DockPanel>
|
</DockPanel>
|
||||||
|
|||||||
@@ -76,13 +76,12 @@ namespace v2rayN.Views
|
|||||||
//fill fonts
|
//fill fonts
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var dir = new DirectoryInfo(Utils.GetFontsPath());
|
var files = Directory.GetFiles(Utils.GetFontsPath(), "*.ttf");
|
||||||
var files = dir.GetFiles("*.ttf");
|
var culture = _config.uiItem.currentLanguage == Global.Languages[0] ? "zh-cn" : "en-us";
|
||||||
var culture = _config.uiItem.currentLanguage.Equals(Global.Languages[0]) ? "zh-cn" : "en-us";
|
|
||||||
var culture2 = "en-us";
|
var culture2 = "en-us";
|
||||||
foreach (var it in files)
|
foreach (var ttf in files)
|
||||||
{
|
{
|
||||||
var families = Fonts.GetFontFamilies(Utils.GetFontsPath(it.Name));
|
var families = Fonts.GetFontFamilies(Utils.GetFontsPath(ttf));
|
||||||
foreach (FontFamily family in families)
|
foreach (FontFamily family in families)
|
||||||
{
|
{
|
||||||
var typefaces = family.GetTypefaces();
|
var typefaces = family.GetTypefaces();
|
||||||
@@ -165,6 +164,7 @@ namespace v2rayN.Views
|
|||||||
this.Bind(ViewModel, vm => vm.currentFontFamily, v => v.cmbcurrentFontFamily.Text).DisposeWith(disposables);
|
this.Bind(ViewModel, vm => vm.currentFontFamily, v => v.cmbcurrentFontFamily.Text).DisposeWith(disposables);
|
||||||
this.Bind(ViewModel, vm => vm.SpeedTestTimeout, v => v.cmbSpeedTestTimeout.Text).DisposeWith(disposables);
|
this.Bind(ViewModel, vm => vm.SpeedTestTimeout, v => v.cmbSpeedTestTimeout.Text).DisposeWith(disposables);
|
||||||
this.Bind(ViewModel, vm => vm.SpeedTestUrl, v => v.cmbSpeedTestUrl.Text).DisposeWith(disposables);
|
this.Bind(ViewModel, vm => vm.SpeedTestUrl, v => v.cmbSpeedTestUrl.Text).DisposeWith(disposables);
|
||||||
|
this.Bind(ViewModel, vm => vm.EnableHWA, v => v.togEnableHWA.IsChecked).DisposeWith(disposables);
|
||||||
|
|
||||||
|
|
||||||
this.Bind(ViewModel, vm => vm.systemProxyAdvancedProtocol, v => v.cmbsystemProxyAdvancedProtocol.Text).DisposeWith(disposables);
|
this.Bind(ViewModel, vm => vm.systemProxyAdvancedProtocol, v => v.cmbsystemProxyAdvancedProtocol.Text).DisposeWith(disposables);
|
||||||
@@ -182,8 +182,10 @@ namespace v2rayN.Views
|
|||||||
this.OneWayBind(ViewModel, vm => vm.TunBypassMode2, v => v.gridTunModeProxy.Visibility, vmToViewConverterOverride: new BooleanToVisibilityTypeConverter()).DisposeWith(disposables);
|
this.OneWayBind(ViewModel, vm => vm.TunBypassMode2, v => v.gridTunModeProxy.Visibility, vmToViewConverterOverride: new BooleanToVisibilityTypeConverter()).DisposeWith(disposables);
|
||||||
this.Bind(ViewModel, vm => vm.TunDirectIP, v => v.txtDirectIP.Text).DisposeWith(disposables);
|
this.Bind(ViewModel, vm => vm.TunDirectIP, v => v.txtDirectIP.Text).DisposeWith(disposables);
|
||||||
this.Bind(ViewModel, vm => vm.TunDirectProcess, v => v.txtDirectProcess.Text).DisposeWith(disposables);
|
this.Bind(ViewModel, vm => vm.TunDirectProcess, v => v.txtDirectProcess.Text).DisposeWith(disposables);
|
||||||
|
this.Bind(ViewModel, vm => vm.TunDirectDNS, v => v.txtDirectDNS.Text).DisposeWith(disposables);
|
||||||
this.Bind(ViewModel, vm => vm.TunProxyIP, v => v.txtProxyIP.Text).DisposeWith(disposables);
|
this.Bind(ViewModel, vm => vm.TunProxyIP, v => v.txtProxyIP.Text).DisposeWith(disposables);
|
||||||
this.Bind(ViewModel, vm => vm.TunProxyProcess, v => v.txtProxyProcess.Text).DisposeWith(disposables);
|
this.Bind(ViewModel, vm => vm.TunProxyProcess, v => v.txtProxyProcess.Text).DisposeWith(disposables);
|
||||||
|
this.Bind(ViewModel, vm => vm.TunProxyDNS, v => v.txtProxyDNS.Text).DisposeWith(disposables);
|
||||||
|
|
||||||
|
|
||||||
this.Bind(ViewModel, vm => vm.CoreType1, v => v.cmbCoreType1.Text).DisposeWith(disposables);
|
this.Bind(ViewModel, vm => vm.CoreType1, v => v.cmbCoreType1.Text).DisposeWith(disposables);
|
||||||
|
|||||||
@@ -134,7 +134,7 @@
|
|||||||
</Grid>
|
</Grid>
|
||||||
|
|
||||||
<Grid
|
<Grid
|
||||||
Margin="16"
|
Margin="8"
|
||||||
HorizontalAlignment="Center"
|
HorizontalAlignment="Center"
|
||||||
DockPanel.Dock="Bottom">
|
DockPanel.Dock="Bottom">
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
|
|||||||
@@ -32,28 +32,40 @@
|
|||||||
Style="{StaticResource MaterialDesignToolBar}">
|
Style="{StaticResource MaterialDesignToolBar}">
|
||||||
<Button x:Name="menuRuleAdd">
|
<Button x:Name="menuRuleAdd">
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="Plus" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="Plus" />
|
||||||
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuRuleAdd}" />
|
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuRuleAdd}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</Button>
|
</Button>
|
||||||
<Separator />
|
<Separator />
|
||||||
<Button x:Name="menuImportRulesFromFile">
|
<Button x:Name="menuImportRulesFromFile">
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="Import" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="Import" />
|
||||||
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuImportRulesFromFile}" />
|
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuImportRulesFromFile}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</Button>
|
</Button>
|
||||||
<Separator />
|
<Separator />
|
||||||
<Button x:Name="menuImportRulesFromClipboard">
|
<Button x:Name="menuImportRulesFromClipboard">
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="Import" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="Import" />
|
||||||
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuImportRulesFromClipboard}" />
|
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuImportRulesFromClipboard}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</Button>
|
</Button>
|
||||||
<Separator />
|
<Separator />
|
||||||
<Button x:Name="menuImportRulesFromUrl">
|
<Button x:Name="menuImportRulesFromUrl">
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="Import" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="Import" />
|
||||||
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuImportRulesFromUrl}" />
|
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuImportRulesFromUrl}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</Button>
|
</Button>
|
||||||
@@ -61,7 +73,7 @@
|
|||||||
</ToolBarTray>
|
</ToolBarTray>
|
||||||
|
|
||||||
<Grid
|
<Grid
|
||||||
Margin="16"
|
Margin="8"
|
||||||
HorizontalAlignment="Center"
|
HorizontalAlignment="Center"
|
||||||
DockPanel.Dock="Bottom">
|
DockPanel.Dock="Bottom">
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
|
|||||||
@@ -39,7 +39,10 @@
|
|||||||
<MenuItem x:Name="menuRoutingBasic" Padding="8,0">
|
<MenuItem x:Name="menuRoutingBasic" Padding="8,0">
|
||||||
<MenuItem.Header>
|
<MenuItem.Header>
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="Server" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="Server" />
|
||||||
<TextBlock Text="{x:Static resx:ResUI.menuRoutingBasic}" />
|
<TextBlock Text="{x:Static resx:ResUI.menuRoutingBasic}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</MenuItem.Header>
|
</MenuItem.Header>
|
||||||
@@ -54,7 +57,10 @@
|
|||||||
<MenuItem x:Name="menuRoutingAdvanced" Padding="8,0">
|
<MenuItem x:Name="menuRoutingAdvanced" Padding="8,0">
|
||||||
<MenuItem.Header>
|
<MenuItem.Header>
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="Routes" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="Routes" />
|
||||||
<TextBlock Text="{x:Static resx:ResUI.menuRoutingAdvanced}" />
|
<TextBlock Text="{x:Static resx:ResUI.menuRoutingAdvanced}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</MenuItem.Header>
|
</MenuItem.Header>
|
||||||
@@ -108,7 +114,7 @@
|
|||||||
</ToolBarTray>
|
</ToolBarTray>
|
||||||
|
|
||||||
<Grid
|
<Grid
|
||||||
Margin="16"
|
Margin="8"
|
||||||
HorizontalAlignment="Center"
|
HorizontalAlignment="Center"
|
||||||
DockPanel.Dock="Bottom">
|
DockPanel.Dock="Bottom">
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
|
|||||||
@@ -162,7 +162,7 @@
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid
|
<Grid
|
||||||
Grid.Row="1"
|
Grid.Row="1"
|
||||||
Margin="16"
|
Margin="8"
|
||||||
HorizontalAlignment="Center">
|
HorizontalAlignment="Center">
|
||||||
<Grid.ColumnDefinitions>
|
<Grid.ColumnDefinitions>
|
||||||
<ColumnDefinition Width="200" />
|
<ColumnDefinition Width="200" />
|
||||||
|
|||||||
@@ -33,28 +33,40 @@
|
|||||||
Style="{StaticResource MaterialDesignToolBar}">
|
Style="{StaticResource MaterialDesignToolBar}">
|
||||||
<Button x:Name="menuSubAdd">
|
<Button x:Name="menuSubAdd">
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="Plus" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="Plus" />
|
||||||
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuSubAdd}" />
|
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuSubAdd}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</Button>
|
</Button>
|
||||||
<Separator />
|
<Separator />
|
||||||
<Button x:Name="menuSubDelete">
|
<Button x:Name="menuSubDelete">
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="Delete" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="Delete" />
|
||||||
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuSubDelete}" />
|
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuSubDelete}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</Button>
|
</Button>
|
||||||
<Separator />
|
<Separator />
|
||||||
<Button x:Name="menuSubEdit">
|
<Button x:Name="menuSubEdit">
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="Edit" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="Edit" />
|
||||||
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuSubEdit}" />
|
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuSubEdit}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</Button>
|
</Button>
|
||||||
<Separator />
|
<Separator />
|
||||||
<Button x:Name="menuSubShare">
|
<Button x:Name="menuSubShare">
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="QrcodePlus" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="QrcodePlus" />
|
||||||
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuSubShare}" />
|
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuSubShare}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</Button>
|
</Button>
|
||||||
@@ -63,7 +75,10 @@
|
|||||||
Click="menuClose_Click"
|
Click="menuClose_Click"
|
||||||
IsCancel="True">
|
IsCancel="True">
|
||||||
<StackPanel Orientation="Horizontal">
|
<StackPanel Orientation="Horizontal">
|
||||||
<materialDesign:PackIcon Margin="0,0,8,0" Kind="Close" />
|
<materialDesign:PackIcon
|
||||||
|
Margin="0,0,8,0"
|
||||||
|
VerticalAlignment="Center"
|
||||||
|
Kind="Close" />
|
||||||
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuClose}" />
|
<TextBlock Style="{StaticResource ToolbarTextBlock}" Text="{x:Static resx:ResUI.menuClose}" />
|
||||||
</StackPanel>
|
</StackPanel>
|
||||||
</Button>
|
</Button>
|
||||||
|
|||||||
@@ -6,4 +6,18 @@
|
|||||||
<dpiAwareness xmlns="http://schemas.microsoft.com/SMI/2016/WindowsSettings">PerMonitorV2</dpiAwareness>
|
<dpiAwareness xmlns="http://schemas.microsoft.com/SMI/2016/WindowsSettings">PerMonitorV2</dpiAwareness>
|
||||||
</asmv3:windowsSettings>
|
</asmv3:windowsSettings>
|
||||||
</asmv3:application>
|
</asmv3:application>
|
||||||
|
|
||||||
|
<!-- Enable themes for Windows common controls and dialogs (Windows XP and later) -->
|
||||||
|
<dependency>
|
||||||
|
<dependentAssembly>
|
||||||
|
<assemblyIdentity
|
||||||
|
type="win32"
|
||||||
|
name="Microsoft.Windows.Common-Controls"
|
||||||
|
version="6.0.0.0"
|
||||||
|
processorArchitecture="*"
|
||||||
|
publicKeyToken="6595b64144ccf1df"
|
||||||
|
language="*"
|
||||||
|
/>
|
||||||
|
</dependentAssembly>
|
||||||
|
</dependency>
|
||||||
</assembly>
|
</assembly>
|
||||||
@@ -10,20 +10,18 @@
|
|||||||
<ImplicitUsings>enable</ImplicitUsings>
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
<ApplicationIcon>v2rayN.ico</ApplicationIcon>
|
<ApplicationIcon>v2rayN.ico</ApplicationIcon>
|
||||||
<Copyright>Copyright © 2017-2023 (GPLv3)</Copyright>
|
<Copyright>Copyright © 2017-2023 (GPLv3)</Copyright>
|
||||||
<FileVersion>6.13</FileVersion>
|
<FileVersion>6.20</FileVersion>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="Downloader" Version="3.0.3" />
|
<PackageReference Include="Downloader" Version="3.0.4" />
|
||||||
<PackageReference Include="MaterialDesignThemes" Version="4.7.1" />
|
<PackageReference Include="MaterialDesignThemes" Version="4.7.1" />
|
||||||
<PackageReference Include="Hardcodet.NotifyIcon.Wpf" Version="1.1.0" />
|
<PackageReference Include="Hardcodet.NotifyIcon.Wpf" Version="1.1.0" />
|
||||||
<PackageReference Include="Newtonsoft.Json" Version="13.0.2" />
|
<PackageReference Include="Newtonsoft.Json" Version="13.0.3" />
|
||||||
<PackageReference Include="NHotkey" Version="2.1.0" />
|
|
||||||
<PackageReference Include="NHotkey.Wpf" Version="2.1.0" />
|
|
||||||
<PackageReference Include="QRCoder.Xaml" Version="1.4.3" />
|
<PackageReference Include="QRCoder.Xaml" Version="1.4.3" />
|
||||||
<PackageReference Include="sqlite-net-pcl" Version="1.8.116" />
|
<PackageReference Include="sqlite-net-pcl" Version="1.8.116" />
|
||||||
<PackageReference Include="TaskScheduler" Version="2.10.1" />
|
<PackageReference Include="TaskScheduler" Version="2.10.1" />
|
||||||
<PackageReference Include="ZXing.Net.Bindings.Windows.Compatibility" Version="0.16.10" />
|
<PackageReference Include="ZXing.Net.Bindings.Windows.Compatibility" Version="0.16.12" />
|
||||||
<PackageReference Include="ReactiveUI.Fody" Version="18.4.1" />
|
<PackageReference Include="ReactiveUI.Fody" Version="18.4.1" />
|
||||||
<PackageReference Include="ReactiveUI.Validation" Version="3.0.22" />
|
<PackageReference Include="ReactiveUI.Validation" Version="3.0.22" />
|
||||||
<PackageReference Include="ReactiveUI.WPF" Version="18.4.1" />
|
<PackageReference Include="ReactiveUI.WPF" Version="18.4.1" />
|
||||||
@@ -36,6 +34,9 @@
|
|||||||
<EmbeddedResource Include="Sample\tun_singbox">
|
<EmbeddedResource Include="Sample\tun_singbox">
|
||||||
<CopyToOutputDirectory>Never</CopyToOutputDirectory>
|
<CopyToOutputDirectory>Never</CopyToOutputDirectory>
|
||||||
</EmbeddedResource>
|
</EmbeddedResource>
|
||||||
|
<EmbeddedResource Include="Sample\tun_singbox_dns">
|
||||||
|
<CopyToOutputDirectory>Never</CopyToOutputDirectory>
|
||||||
|
</EmbeddedResource>
|
||||||
<EmbeddedResource Include="Sample\custom_routing_black">
|
<EmbeddedResource Include="Sample\custom_routing_black">
|
||||||
<CopyToOutputDirectory>Never</CopyToOutputDirectory>
|
<CopyToOutputDirectory>Never</CopyToOutputDirectory>
|
||||||
</EmbeddedResource>
|
</EmbeddedResource>
|
||||||
@@ -65,7 +66,7 @@
|
|||||||
</EmbeddedResource>
|
</EmbeddedResource>
|
||||||
<EmbeddedResource Include="Sample\SampleServerConfig">
|
<EmbeddedResource Include="Sample\SampleServerConfig">
|
||||||
<CopyToOutputDirectory>Never</CopyToOutputDirectory>
|
<CopyToOutputDirectory>Never</CopyToOutputDirectory>
|
||||||
</EmbeddedResource>
|
</EmbeddedResource>
|
||||||
<EmbeddedResource Include="v2rayN.ico">
|
<EmbeddedResource Include="v2rayN.ico">
|
||||||
<CopyToOutputDirectory>Never</CopyToOutputDirectory>
|
<CopyToOutputDirectory>Never</CopyToOutputDirectory>
|
||||||
</EmbeddedResource>
|
</EmbeddedResource>
|
||||||
|
|||||||
@@ -11,18 +11,22 @@ namespace v2rayUpgrade
|
|||||||
public partial class MainForm : Form
|
public partial class MainForm : Form
|
||||||
{
|
{
|
||||||
private readonly string defaultFilename = "v2ray-windows.zip";
|
private readonly string defaultFilename = "v2ray-windows.zip";
|
||||||
private string fileName;
|
private string? fileName;
|
||||||
|
|
||||||
public MainForm(string[] args)
|
public MainForm(string[] args)
|
||||||
{
|
{
|
||||||
InitializeComponent();
|
InitializeComponent();
|
||||||
if (args.Length > 0)
|
if (args.Length > 0)
|
||||||
{
|
{
|
||||||
fileName = string.Join(" ", args);
|
fileName = HttpUtility.UrlDecode(string.Join(" ", args));
|
||||||
fileName = HttpUtility.UrlDecode(fileName);
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
fileName = defaultFilename;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
private void showWarn(string message)
|
|
||||||
|
private void ShowWarn(string message)
|
||||||
{
|
{
|
||||||
MessageBox.Show(message, "", MessageBoxButtons.OK, MessageBoxIcon.Warning);
|
MessageBox.Show(message, "", MessageBoxButtons.OK, MessageBoxIcon.Warning);
|
||||||
}
|
}
|
||||||
@@ -34,7 +38,7 @@ namespace v2rayUpgrade
|
|||||||
Process[] existing = Process.GetProcessesByName("v2rayN");
|
Process[] existing = Process.GetProcessesByName("v2rayN");
|
||||||
foreach (Process p in existing)
|
foreach (Process p in existing)
|
||||||
{
|
{
|
||||||
string path = p.MainModule.FileName;
|
string? path = p.MainModule?.FileName;
|
||||||
if (path == GetPath("v2rayN.exe"))
|
if (path == GetPath("v2rayN.exe"))
|
||||||
{
|
{
|
||||||
p.Kill();
|
p.Kill();
|
||||||
@@ -45,72 +49,67 @@ namespace v2rayUpgrade
|
|||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
// Access may be denied without admin right. The user may not be an administrator.
|
// Access may be denied without admin right. The user may not be an administrator.
|
||||||
showWarn("Failed to close v2rayN(关闭v2rayN失败).\n" +
|
ShowWarn("Failed to close v2rayN(关闭v2rayN失败).\n" +
|
||||||
"Close it manually, or the upgrade may fail.(请手动关闭正在运行的v2rayN,否则可能升级失败。\n\n" + ex.StackTrace);
|
"Close it manually, or the upgrade may fail.(请手动关闭正在运行的v2rayN,否则可能升级失败。\n\n" + ex.StackTrace);
|
||||||
}
|
}
|
||||||
|
|
||||||
StringBuilder sb = new StringBuilder();
|
if (!File.Exists(fileName))
|
||||||
|
{
|
||||||
|
if (File.Exists(defaultFilename))
|
||||||
|
{
|
||||||
|
fileName = defaultFilename;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ShowWarn("Upgrade Failed, File Not Exist(升级失败,文件不存在).");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
StringBuilder sb = new();
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
if (!File.Exists(fileName))
|
string thisAppOldFile = $"{Application.ExecutablePath}.tmp";
|
||||||
{
|
|
||||||
if (File.Exists(defaultFilename))
|
|
||||||
{
|
|
||||||
fileName = defaultFilename;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
showWarn("Upgrade Failed, File Not Exist(升级失败,文件不存在).");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
string thisAppOldFile = Application.ExecutablePath + ".tmp";
|
|
||||||
File.Delete(thisAppOldFile);
|
File.Delete(thisAppOldFile);
|
||||||
string startKey = "v2rayN/";
|
string startKey = "v2rayN/";
|
||||||
|
|
||||||
|
using ZipArchive archive = ZipFile.OpenRead(fileName);
|
||||||
using (ZipArchive archive = ZipFile.OpenRead(fileName))
|
foreach (ZipArchiveEntry entry in archive.Entries)
|
||||||
{
|
{
|
||||||
foreach (ZipArchiveEntry entry in archive.Entries)
|
try
|
||||||
{
|
{
|
||||||
try
|
if (entry.Length == 0)
|
||||||
{
|
{
|
||||||
if (entry.Length == 0)
|
continue;
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
string fullName = entry.FullName;
|
|
||||||
if (fullName.StartsWith(startKey))
|
|
||||||
{
|
|
||||||
fullName = fullName.Substring(startKey.Length, fullName.Length - startKey.Length);
|
|
||||||
}
|
|
||||||
if (Application.ExecutablePath.ToLower() == GetPath(fullName).ToLower())
|
|
||||||
{
|
|
||||||
File.Move(Application.ExecutablePath, thisAppOldFile);
|
|
||||||
}
|
|
||||||
|
|
||||||
string entryOuputPath = GetPath(fullName);
|
|
||||||
|
|
||||||
FileInfo fileInfo = new FileInfo(entryOuputPath);
|
|
||||||
fileInfo.Directory.Create();
|
|
||||||
entry.ExtractToFile(entryOuputPath, true);
|
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
string fullName = entry.FullName;
|
||||||
|
if (fullName.StartsWith(startKey))
|
||||||
{
|
{
|
||||||
sb.Append(ex.StackTrace);
|
fullName = fullName[startKey.Length..];
|
||||||
}
|
}
|
||||||
|
if (string.Equals(Application.ExecutablePath, GetPath(fullName), StringComparison.OrdinalIgnoreCase))
|
||||||
|
{
|
||||||
|
File.Move(Application.ExecutablePath, thisAppOldFile);
|
||||||
|
}
|
||||||
|
|
||||||
|
string entryOuputPath = GetPath(fullName);
|
||||||
|
Directory.CreateDirectory(Path.GetDirectoryName(entryOuputPath)!);
|
||||||
|
entry.ExtractToFile(entryOuputPath, true);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
sb.Append(ex.StackTrace);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
showWarn("Upgrade Failed(升级失败)." + ex.StackTrace);
|
ShowWarn("Upgrade Failed(升级失败)." + ex.StackTrace);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (sb.Length > 0)
|
if (sb.Length > 0)
|
||||||
{
|
{
|
||||||
showWarn("Upgrade Failed,Hold ctrl + c to copy to clipboard.\n" +
|
ShowWarn("Upgrade Failed,Hold ctrl + c to copy to clipboard.\n" +
|
||||||
"(升级失败,按住ctrl+c可以复制到剪贴板)." + sb.ToString());
|
"(升级失败,按住ctrl+c可以复制到剪贴板)." + sb.ToString());
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,5 +6,6 @@
|
|||||||
<Copyright>Copyright © 2019-2023 (GPLv3)</Copyright>
|
<Copyright>Copyright © 2019-2023 (GPLv3)</Copyright>
|
||||||
<FileVersion>1.1.0.0</FileVersion>
|
<FileVersion>1.1.0.0</FileVersion>
|
||||||
<ApplicationManifest>app.manifest</ApplicationManifest>
|
<ApplicationManifest>app.manifest</ApplicationManifest>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
</Project>
|
</Project>
|
||||||
Reference in New Issue
Block a user