mirror of
				https://github.com/2dust/v2rayN.git
				synced 2025-10-25 17:54:41 +00:00 
			
		
		
		
	Compare commits
	
		
			No commits in common. "d8b0363bb79bb1f6ce7da5e28992709df0b55508" and "e3a90442a91d5a26b653dfb8563a29a951712998" have entirely different histories.
		
	
	
		
			d8b0363bb7
			...
			e3a90442a9
		
	
		
					 24 changed files with 212 additions and 128 deletions
				
			
		
							
								
								
									
										31
									
								
								.github/workflows/winget-publish.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										31
									
								
								.github/workflows/winget-publish.yml
									
									
									
									
										vendored
									
									
								
							|  | @ -1,31 +0,0 @@ | ||||||
| name: WinGet submission on release |  | ||||||
| # based off of https://github.com/nushell/nushell/blob/main/.github/workflows/winget-submission.yml |  | ||||||
| # inspired by https://github.com/microsoft/PowerToys/blob/main/.github/workflows/package-submissions.yml |  | ||||||
| # Modified by @MerrickZ https://github.com/anpho |  | ||||||
| 
 |  | ||||||
| on: |  | ||||||
|   workflow_dispatch: |  | ||||||
|   release: |  | ||||||
|     types: [released] |  | ||||||
| 
 |  | ||||||
| jobs: |  | ||||||
|   winget: |  | ||||||
|     name: Publish winget package |  | ||||||
|     runs-on: windows-latest |  | ||||||
|     steps: |  | ||||||
|       - name: Submit v2ray package to Windows Package Manager Community Repository |  | ||||||
|         run: | |  | ||||||
| 
 |  | ||||||
|           $wingetPackage = "2dust.v2rayN" |  | ||||||
|           $gitToken = "${{ secrets.PT_WINGET }}" |  | ||||||
| 
 |  | ||||||
|           $github = Invoke-RestMethod -uri "https://api.github.com/repos/2dust/v2rayN/releases"  |  | ||||||
| 
 |  | ||||||
|           $targetRelease = $github | Where-Object -Property prerelease -match 'False' | Select -First 1 |  | ||||||
|           $installerUrl  = $targetRelease | Select -ExpandProperty assets -First 1 | Where-Object -Property name -match 'v2rayN-windows-64-With-Core\.zip*' | Select -ExpandProperty browser_download_url |  | ||||||
| 
 |  | ||||||
|           $ver = $targetRelease.tag_name |  | ||||||
| 
 |  | ||||||
|           # getting latest wingetcreate file |  | ||||||
|           iwr https://aka.ms/wingetcreate/latest -OutFile wingetcreate.exe |  | ||||||
|           .\wingetcreate.exe update $wingetPackage -s -v $ver -u "$installerUrl|x64" -t $gitToken |  | ||||||
|  | @ -31,8 +31,8 @@ | ||||||
|                 var parts = this.version.Split('.'); |                 var parts = this.version.Split('.'); | ||||||
|                 if (parts.Length == 2) |                 if (parts.Length == 2) | ||||||
|                 { |                 { | ||||||
|                     this.major = int.Parse(parts.First()); |                     this.major = int.Parse(parts[0]); | ||||||
|                     this.minor = int.Parse(parts.Last()); |                     this.minor = int.Parse(parts[1]); | ||||||
|                     this.patch = 0; |                     this.patch = 0; | ||||||
|                 } |                 } | ||||||
|                 else if (parts.Length is 3 or 4) |                 else if (parts.Length is 3 or 4) | ||||||
|  |  | ||||||
|  | @ -22,7 +22,7 @@ namespace ServiceLib.Common | ||||||
|         public static bool BeginWithAny(this string s, IEnumerable<char> chars) |         public static bool BeginWithAny(this string s, IEnumerable<char> chars) | ||||||
|         { |         { | ||||||
|             if (s.IsNullOrEmpty()) return false; |             if (s.IsNullOrEmpty()) return false; | ||||||
|             return chars.Contains(s.First()); |             return chars.Contains(s[0]); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         private static bool IsWhiteSpace(this string value) |         private static bool IsWhiteSpace(this string value) | ||||||
|  | @ -61,7 +61,7 @@ namespace ServiceLib.Common | ||||||
|                 return string.Empty; |                 return string.Empty; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             return char.ToUpper(value.First()) + value[1..]; |             return char.ToUpper(value[0]) + value[1..]; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         public static string AppendQuotes(this string value) |         public static string AppendQuotes(this string value) | ||||||
|  |  | ||||||
|  | @ -313,8 +313,8 @@ namespace ServiceLib.Common | ||||||
|                     continue; |                     continue; | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 var key = Uri.UnescapeDataString(keyValue.First()); |                 var key = Uri.UnescapeDataString(keyValue[0]); | ||||||
|                 var val = Uri.UnescapeDataString(keyValue.Last()); |                 var val = Uri.UnescapeDataString(keyValue[1]); | ||||||
| 
 | 
 | ||||||
|                 if (result[key] is null) |                 if (result[key] is null) | ||||||
|                 { |                 { | ||||||
|  | @ -622,8 +622,8 @@ namespace ServiceLib.Common | ||||||
|                     { |                     { | ||||||
|                         if (host.StartsWith("#")) continue; |                         if (host.StartsWith("#")) continue; | ||||||
|                         var hostItem = host.Split(new[] { ' ', '\t' }, StringSplitOptions.RemoveEmptyEntries); |                         var hostItem = host.Split(new[] { ' ', '\t' }, StringSplitOptions.RemoveEmptyEntries); | ||||||
|                         if (hostItem.Length != 2) continue; |                         if (hostItem.Length < 2) continue; | ||||||
|                         systemHosts.Add(hostItem.Last(), hostItem.First()); |                         systemHosts.Add(hostItem[1], hostItem[0]); | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |  | ||||||
|  | @ -62,7 +62,7 @@ namespace ServiceLib.Handler | ||||||
|             { |             { | ||||||
|                 if (config.Inbound.Count > 0) |                 if (config.Inbound.Count > 0) | ||||||
|                 { |                 { | ||||||
|                     config.Inbound.First().Protocol = EInboundProtocol.socks.ToString(); |                     config.Inbound[0].Protocol = EInboundProtocol.socks.ToString(); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  | @ -70,7 +70,7 @@ namespace ServiceLib.Handler | ||||||
| 
 | 
 | ||||||
|             if (Utils.IsNullOrEmpty(config.RoutingBasicItem.DomainStrategy)) |             if (Utils.IsNullOrEmpty(config.RoutingBasicItem.DomainStrategy)) | ||||||
|             { |             { | ||||||
|                 config.RoutingBasicItem.DomainStrategy = Global.DomainStrategies.First();//"IPIfNonMatch"; |                 config.RoutingBasicItem.DomainStrategy = Global.DomainStrategies[0];//"IPIfNonMatch"; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             config.KcpItem ??= new KcpItem |             config.KcpItem ??= new KcpItem | ||||||
|  | @ -111,7 +111,7 @@ namespace ServiceLib.Handler | ||||||
|             { |             { | ||||||
|                 if (Thread.CurrentThread.CurrentCulture.Name.Equals("zh-cn", StringComparison.CurrentCultureIgnoreCase)) |                 if (Thread.CurrentThread.CurrentCulture.Name.Equals("zh-cn", StringComparison.CurrentCultureIgnoreCase)) | ||||||
|                 { |                 { | ||||||
|                     config.UiItem.CurrentLanguage = Global.Languages.First(); |                     config.UiItem.CurrentLanguage = Global.Languages[0]; | ||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|                 { |                 { | ||||||
|  | @ -132,7 +132,7 @@ namespace ServiceLib.Handler | ||||||
|             } |             } | ||||||
|             if (Utils.IsNullOrEmpty(config.SpeedTestItem.SpeedTestUrl)) |             if (Utils.IsNullOrEmpty(config.SpeedTestItem.SpeedTestUrl)) | ||||||
|             { |             { | ||||||
|                 config.SpeedTestItem.SpeedTestUrl = Global.SpeedTestUrls.First(); |                 config.SpeedTestItem.SpeedTestUrl = Global.SpeedTestUrls[0]; | ||||||
|             } |             } | ||||||
|             if (Utils.IsNullOrEmpty(config.SpeedTestItem.SpeedPingTestUrl)) |             if (Utils.IsNullOrEmpty(config.SpeedTestItem.SpeedPingTestUrl)) | ||||||
|             { |             { | ||||||
|  | @ -148,7 +148,7 @@ namespace ServiceLib.Handler | ||||||
| 
 | 
 | ||||||
|             config.Mux4SboxItem ??= new() |             config.Mux4SboxItem ??= new() | ||||||
|             { |             { | ||||||
|                 Protocol = Global.SingboxMuxs.First(), |                 Protocol = Global.SingboxMuxs[0], | ||||||
|                 MaxConnections = 8 |                 MaxConnections = 8 | ||||||
|             }; |             }; | ||||||
| 
 | 
 | ||||||
|  | @ -429,7 +429,7 @@ namespace ServiceLib.Handler | ||||||
|                         { |                         { | ||||||
|                             return 0; |                             return 0; | ||||||
|                         } |                         } | ||||||
|                         sort = ProfileExHandler.Instance.GetSort(lstProfile.First().IndexId) - 1; |                         sort = ProfileExHandler.Instance.GetSort(lstProfile[0].IndexId) - 1; | ||||||
| 
 | 
 | ||||||
|                         break; |                         break; | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
|  | @ -99,8 +99,8 @@ namespace ServiceLib.Handler.Fmt | ||||||
|                 { |                 { | ||||||
|                     return null; |                     return null; | ||||||
|                 } |                 } | ||||||
|                 item.Security = userInfoParts.First(); |                 item.Security = userInfoParts[0]; | ||||||
|                 item.Id = Utils.UrlDecode(userInfoParts.Last()); |                 item.Id = Utils.UrlDecode(userInfoParts[1]); | ||||||
|             } |             } | ||||||
|             else |             else | ||||||
|             { |             { | ||||||
|  | @ -111,8 +111,8 @@ namespace ServiceLib.Handler.Fmt | ||||||
|                 { |                 { | ||||||
|                     return null; |                     return null; | ||||||
|                 } |                 } | ||||||
|                 item.Security = userInfoParts.First(); |                 item.Security = userInfoParts[0]; | ||||||
|                 item.Id = userInfoParts.Last(); |                 item.Id = userInfoParts[1]; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             var queryParameters = Utils.ParseQueryString(parsedUrl.Query); |             var queryParameters = Utils.ParseQueryString(parsedUrl.Query); | ||||||
|  |  | ||||||
|  | @ -5,8 +5,9 @@ | ||||||
|         public static ProfileItem? Resolve(string str, out string msg) |         public static ProfileItem? Resolve(string str, out string msg) | ||||||
|         { |         { | ||||||
|             msg = ResUI.ConfigurationFormatIncorrect; |             msg = ResUI.ConfigurationFormatIncorrect; | ||||||
|  |             ProfileItem? item; | ||||||
| 
 | 
 | ||||||
|             var item = ResolveSocksNew(str) ?? ResolveSocks(str); |             item = ResolveSocksNew(str) ?? ResolveSocks(str); | ||||||
|             if (item == null) |             if (item == null) | ||||||
|             { |             { | ||||||
|                 return null; |                 return null; | ||||||
|  | @ -24,13 +25,19 @@ | ||||||
|         public static string? ToUri(ProfileItem? item) |         public static string? ToUri(ProfileItem? item) | ||||||
|         { |         { | ||||||
|             if (item == null) return null; |             if (item == null) return null; | ||||||
|             var url = string.Empty; |             string url = string.Empty; | ||||||
| 
 | 
 | ||||||
|             var remark = string.Empty; |             string remark = string.Empty; | ||||||
|             if (Utils.IsNotEmpty(item.Remarks)) |             if (Utils.IsNotEmpty(item.Remarks)) | ||||||
|             { |             { | ||||||
|                 remark = "#" + Utils.UrlEncode(item.Remarks); |                 remark = "#" + Utils.UrlEncode(item.Remarks); | ||||||
|             } |             } | ||||||
|  |             //url = string.Format("{0}:{1}@{2}:{3}", | ||||||
|  |             //    item.security, | ||||||
|  |             //    item.id, | ||||||
|  |             //    item.address, | ||||||
|  |             //    item.port); | ||||||
|  |             //url = Utile.Base64Encode(url); | ||||||
|             //new |             //new | ||||||
|             var pw = Utils.Base64Encode($"{item.Security}:{item.Id}"); |             var pw = Utils.Base64Encode($"{item.Security}:{item.Id}"); | ||||||
|             return ToUri(EConfigType.SOCKS, item.Address, item.Port, pw, null, remark); |             return ToUri(EConfigType.SOCKS, item.Address, item.Port, pw, null, remark); | ||||||
|  | @ -44,7 +51,7 @@ | ||||||
|             }; |             }; | ||||||
|             result = result[Global.ProtocolShares[EConfigType.SOCKS].Length..]; |             result = result[Global.ProtocolShares[EConfigType.SOCKS].Length..]; | ||||||
|             //remark |             //remark | ||||||
|             var indexRemark = result.IndexOf("#"); |             int indexRemark = result.IndexOf("#"); | ||||||
|             if (indexRemark > 0) |             if (indexRemark > 0) | ||||||
|             { |             { | ||||||
|                 try |                 try | ||||||
|  | @ -55,7 +62,7 @@ | ||||||
|                 result = result[..indexRemark]; |                 result = result[..indexRemark]; | ||||||
|             } |             } | ||||||
|             //part decode |             //part decode | ||||||
|             var indexS = result.IndexOf("@"); |             int indexS = result.IndexOf("@"); | ||||||
|             if (indexS > 0) |             if (indexS > 0) | ||||||
|             { |             { | ||||||
|             } |             } | ||||||
|  | @ -64,20 +71,21 @@ | ||||||
|                 result = Utils.Base64Decode(result); |                 result = Utils.Base64Decode(result); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             var arr1 = result.Split('@'); |             string[] arr1 = result.Split('@'); | ||||||
|             if (arr1.Length != 2) |             if (arr1.Length != 2) | ||||||
|             { |             { | ||||||
|                 return null; |                 return null; | ||||||
|             } |             } | ||||||
|             var arr21 = arr1.First().Split(':'); |             string[] arr21 = arr1[0].Split(':'); | ||||||
|             var indexPort = arr1.Last().LastIndexOf(":"); |             //string[] arr22 = arr1[1].Split(':'); | ||||||
|  |             int indexPort = arr1[1].LastIndexOf(":"); | ||||||
|             if (arr21.Length != 2 || indexPort < 0) |             if (arr21.Length != 2 || indexPort < 0) | ||||||
|             { |             { | ||||||
|                 return null; |                 return null; | ||||||
|             } |             } | ||||||
|             item.Address = arr1[1][..indexPort]; |             item.Address = arr1[1][..indexPort]; | ||||||
|             item.Port = Utils.ToInt(arr1[1][(indexPort + 1)..]); |             item.Port = Utils.ToInt(arr1[1][(indexPort + 1)..]); | ||||||
|             item.Security = arr21.First(); |             item.Security = arr21[0]; | ||||||
|             item.Id = arr21[1]; |             item.Id = arr21[1]; | ||||||
| 
 | 
 | ||||||
|             return item; |             return item; | ||||||
|  | @ -98,10 +106,10 @@ | ||||||
|             // parse base64 UserInfo |             // parse base64 UserInfo | ||||||
|             var rawUserInfo = Utils.UrlDecode(parsedUrl.UserInfo); |             var rawUserInfo = Utils.UrlDecode(parsedUrl.UserInfo); | ||||||
|             var userInfo = Utils.Base64Decode(rawUserInfo); |             var userInfo = Utils.Base64Decode(rawUserInfo); | ||||||
|             var userInfoParts = userInfo.Split([':'], 2); |             var userInfoParts = userInfo.Split(new[] { ':' }, 2); | ||||||
|             if (userInfoParts.Length == 2) |             if (userInfoParts.Length == 2) | ||||||
|             { |             { | ||||||
|                 item.Security = userInfoParts.First(); |                 item.Security = userInfoParts[0]; | ||||||
|                 item.Id = userInfoParts[1]; |                 item.Id = userInfoParts[1]; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -21,8 +21,8 @@ | ||||||
|             var userInfoParts = rawUserInfo.Split(new[] { ':' }, 2); |             var userInfoParts = rawUserInfo.Split(new[] { ':' }, 2); | ||||||
|             if (userInfoParts.Length == 2) |             if (userInfoParts.Length == 2) | ||||||
|             { |             { | ||||||
|                 item.Id = userInfoParts.First(); |                 item.Id = userInfoParts[0]; | ||||||
|                 item.Security = userInfoParts.Last(); |                 item.Security = userInfoParts[1]; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             var query = Utils.ParseQueryString(url.Query); |             var query = Utils.ParseQueryString(url.Query); | ||||||
|  |  | ||||||
|  | @ -222,6 +222,7 @@ | ||||||
|         public int ProxiesSorting { get; set; } |         public int ProxiesSorting { get; set; } | ||||||
|         public bool ProxiesAutoRefresh { get; set; } |         public bool ProxiesAutoRefresh { get; set; } | ||||||
|         public int ProxiesAutoDelayTestInterval { get; set; } = 10; |         public int ProxiesAutoDelayTestInterval { get; set; } = 10; | ||||||
|  |         public int ConnectionsSorting { get; set; } | ||||||
|         public bool ConnectionsAutoRefresh { get; set; } |         public bool ConnectionsAutoRefresh { get; set; } | ||||||
|         public int ConnectionsRefreshInterval { get; set; } = 2; |         public int ConnectionsRefreshInterval { get; set; } = 2; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -88,7 +88,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                 //external-controller |                 //external-controller | ||||||
|                 fileContent["external-controller"] = $"{Global.Loopback}:{AppHandler.Instance.StatePort2}"; |                 fileContent["external-controller"] = $"{Global.Loopback}:{AppHandler.Instance.StatePort2}"; | ||||||
|                 //allow-lan |                 //allow-lan | ||||||
|                 if (_config.Inbound.First().AllowLANConn) |                 if (_config.Inbound[0].AllowLANConn) | ||||||
|                 { |                 { | ||||||
|                     fileContent["allow-lan"] = "true"; |                     fileContent["allow-lan"] = "true"; | ||||||
|                     fileContent["bind-address"] = "*"; |                     fileContent["bind-address"] = "*"; | ||||||
|  |  | ||||||
|  | @ -52,7 +52,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
| 
 | 
 | ||||||
|                 await GenInbounds(singboxConfig); |                 await GenInbounds(singboxConfig); | ||||||
| 
 | 
 | ||||||
|                 await GenOutbound(node, singboxConfig.outbounds.First()); |                 await GenOutbound(node, singboxConfig.outbounds[0]); | ||||||
| 
 | 
 | ||||||
|                 await GenMoreOutbounds(node, singboxConfig); |                 await GenMoreOutbounds(node, singboxConfig); | ||||||
| 
 | 
 | ||||||
|  | @ -495,8 +495,8 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                     singboxConfig.inbounds.Add(inbound); |                     singboxConfig.inbounds.Add(inbound); | ||||||
| 
 | 
 | ||||||
|                     inbound.listen_port = AppHandler.Instance.GetLocalPort(EInboundProtocol.socks); |                     inbound.listen_port = AppHandler.Instance.GetLocalPort(EInboundProtocol.socks); | ||||||
|                     inbound.sniff = _config.Inbound.First().SniffingEnabled; |                     inbound.sniff = _config.Inbound[0].SniffingEnabled; | ||||||
|                     inbound.sniff_override_destination = _config.Inbound.First().RouteOnly ? false : _config.Inbound.First().SniffingEnabled; |                     inbound.sniff_override_destination = _config.Inbound[0].RouteOnly ? false : _config.Inbound[0].SniffingEnabled; | ||||||
|                     inbound.domain_strategy = Utils.IsNullOrEmpty(_config.RoutingBasicItem.DomainStrategy4Singbox) ? null : _config.RoutingBasicItem.DomainStrategy4Singbox; |                     inbound.domain_strategy = Utils.IsNullOrEmpty(_config.RoutingBasicItem.DomainStrategy4Singbox) ? null : _config.RoutingBasicItem.DomainStrategy4Singbox; | ||||||
| 
 | 
 | ||||||
|                     var routing = await ConfigHandler.GetDefaultRouting(_config); |                     var routing = await ConfigHandler.GetDefaultRouting(_config); | ||||||
|  | @ -509,9 +509,9 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                     var inbound2 = GetInbound(inbound, EInboundProtocol.http, false); |                     var inbound2 = GetInbound(inbound, EInboundProtocol.http, false); | ||||||
|                     singboxConfig.inbounds.Add(inbound2); |                     singboxConfig.inbounds.Add(inbound2); | ||||||
| 
 | 
 | ||||||
|                     if (_config.Inbound.First().AllowLANConn) |                     if (_config.Inbound[0].AllowLANConn) | ||||||
|                     { |                     { | ||||||
|                         if (_config.Inbound.First().NewPort4LAN) |                         if (_config.Inbound[0].NewPort4LAN) | ||||||
|                         { |                         { | ||||||
|                             var inbound3 = GetInbound(inbound, EInboundProtocol.socks2, true); |                             var inbound3 = GetInbound(inbound, EInboundProtocol.socks2, true); | ||||||
|                             inbound3.listen = listen; |                             inbound3.listen = listen; | ||||||
|  | @ -522,10 +522,10 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                             singboxConfig.inbounds.Add(inbound4); |                             singboxConfig.inbounds.Add(inbound4); | ||||||
| 
 | 
 | ||||||
|                             //auth |                             //auth | ||||||
|                             if (Utils.IsNotEmpty(_config.Inbound.First().User) && Utils.IsNotEmpty(_config.Inbound.First().Pass)) |                             if (Utils.IsNotEmpty(_config.Inbound[0].User) && Utils.IsNotEmpty(_config.Inbound[0].Pass)) | ||||||
|                             { |                             { | ||||||
|                                 inbound3.users = new() { new() { username = _config.Inbound.First().User, password = _config.Inbound.First().Pass } }; |                                 inbound3.users = new() { new() { username = _config.Inbound[0].User, password = _config.Inbound[0].Pass } }; | ||||||
|                                 inbound4.users = new() { new() { username = _config.Inbound.First().User, password = _config.Inbound.First().Pass } }; |                                 inbound4.users = new() { new() { username = _config.Inbound[0].User, password = _config.Inbound[0].Pass } }; | ||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
|                         else |                         else | ||||||
|  | @ -540,11 +540,11 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                 { |                 { | ||||||
|                     if (_config.TunModeItem.Mtu <= 0) |                     if (_config.TunModeItem.Mtu <= 0) | ||||||
|                     { |                     { | ||||||
|                         _config.TunModeItem.Mtu = Utils.ToInt(Global.TunMtus.First()); |                         _config.TunModeItem.Mtu = Utils.ToInt(Global.TunMtus[0]); | ||||||
|                     } |                     } | ||||||
|                     if (Utils.IsNullOrEmpty(_config.TunModeItem.Stack)) |                     if (Utils.IsNullOrEmpty(_config.TunModeItem.Stack)) | ||||||
|                     { |                     { | ||||||
|                         _config.TunModeItem.Stack = Global.TunStacks.First(); |                         _config.TunModeItem.Stack = Global.TunStacks[0]; | ||||||
|                     } |                     } | ||||||
| 
 | 
 | ||||||
|                     var tunInbound = JsonUtils.Deserialize<Inbound4Sbox>(Utils.GetEmbedText(Global.TunSingboxInboundFileName)) ?? new Inbound4Sbox { }; |                     var tunInbound = JsonUtils.Deserialize<Inbound4Sbox>(Utils.GetEmbedText(Global.TunSingboxInboundFileName)) ?? new Inbound4Sbox { }; | ||||||
|  | @ -552,8 +552,8 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                     tunInbound.mtu = _config.TunModeItem.Mtu; |                     tunInbound.mtu = _config.TunModeItem.Mtu; | ||||||
|                     tunInbound.strict_route = _config.TunModeItem.StrictRoute; |                     tunInbound.strict_route = _config.TunModeItem.StrictRoute; | ||||||
|                     tunInbound.stack = _config.TunModeItem.Stack; |                     tunInbound.stack = _config.TunModeItem.Stack; | ||||||
|                     tunInbound.sniff = _config.Inbound.First().SniffingEnabled; |                     tunInbound.sniff = _config.Inbound[0].SniffingEnabled; | ||||||
|                     //tunInbound.sniff_override_destination = _config.inbound.First().routeOnly ? false : _config.inbound.First().sniffingEnabled; |                     //tunInbound.sniff_override_destination = _config.inbound[0].routeOnly ? false : _config.inbound[0].sniffingEnabled; | ||||||
|                     if (_config.TunModeItem.EnableIPv6Address == false) |                     if (_config.TunModeItem.EnableIPv6Address == false) | ||||||
|                     { |                     { | ||||||
|                         tunInbound.address = ["172.18.0.1/30"]; |                         tunInbound.address = ["172.18.0.1/30"]; | ||||||
|  | @ -867,7 +867,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 //current proxy |                 //current proxy | ||||||
|                 var outbound = singboxConfig.outbounds.First(); |                 var outbound = singboxConfig.outbounds[0]; | ||||||
|                 var txtOutbound = Utils.GetEmbedText(Global.SingboxSampleOutbound); |                 var txtOutbound = Utils.GetEmbedText(Global.SingboxSampleOutbound); | ||||||
| 
 | 
 | ||||||
|                 //Previous proxy |                 //Previous proxy | ||||||
|  | @ -910,7 +910,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|             try |             try | ||||||
|             { |             { | ||||||
|                 var dnsOutbound = "dns_out"; |                 var dnsOutbound = "dns_out"; | ||||||
|                 if (!_config.Inbound.First().SniffingEnabled) |                 if (!_config.Inbound[0].SniffingEnabled) | ||||||
|                 { |                 { | ||||||
|                     singboxConfig.route.rules.Add(new() |                     singboxConfig.route.rules.Add(new() | ||||||
|                     { |                     { | ||||||
|  |  | ||||||
|  | @ -55,7 +55,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
| 
 | 
 | ||||||
|                 await GenRouting(v2rayConfig); |                 await GenRouting(v2rayConfig); | ||||||
| 
 | 
 | ||||||
|                 await GenOutbound(node, v2rayConfig.outbounds.First()); |                 await GenOutbound(node, v2rayConfig.outbounds[0]); | ||||||
| 
 | 
 | ||||||
|                 await GenMoreOutbounds(node, v2rayConfig); |                 await GenMoreOutbounds(node, v2rayConfig); | ||||||
| 
 | 
 | ||||||
|  | @ -391,33 +391,33 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                 var listen = "0.0.0.0"; |                 var listen = "0.0.0.0"; | ||||||
|                 v2rayConfig.inbounds = []; |                 v2rayConfig.inbounds = []; | ||||||
| 
 | 
 | ||||||
|                 Inbounds4Ray? inbound = GetInbound(_config.Inbound.First(), EInboundProtocol.socks, true); |                 Inbounds4Ray? inbound = GetInbound(_config.Inbound[0], EInboundProtocol.socks, true); | ||||||
|                 v2rayConfig.inbounds.Add(inbound); |                 v2rayConfig.inbounds.Add(inbound); | ||||||
| 
 | 
 | ||||||
|                 //http |                 //http | ||||||
|                 Inbounds4Ray? inbound2 = GetInbound(_config.Inbound.First(), EInboundProtocol.http, false); |                 Inbounds4Ray? inbound2 = GetInbound(_config.Inbound[0], EInboundProtocol.http, false); | ||||||
|                 v2rayConfig.inbounds.Add(inbound2); |                 v2rayConfig.inbounds.Add(inbound2); | ||||||
| 
 | 
 | ||||||
|                 if (_config.Inbound.First().AllowLANConn) |                 if (_config.Inbound[0].AllowLANConn) | ||||||
|                 { |                 { | ||||||
|                     if (_config.Inbound.First().NewPort4LAN) |                     if (_config.Inbound[0].NewPort4LAN) | ||||||
|                     { |                     { | ||||||
|                         var inbound3 = GetInbound(_config.Inbound.First(), EInboundProtocol.socks2, true); |                         var inbound3 = GetInbound(_config.Inbound[0], EInboundProtocol.socks2, true); | ||||||
|                         inbound3.listen = listen; |                         inbound3.listen = listen; | ||||||
|                         v2rayConfig.inbounds.Add(inbound3); |                         v2rayConfig.inbounds.Add(inbound3); | ||||||
| 
 | 
 | ||||||
|                         var inbound4 = GetInbound(_config.Inbound.First(), EInboundProtocol.http2, false); |                         var inbound4 = GetInbound(_config.Inbound[0], EInboundProtocol.http2, false); | ||||||
|                         inbound4.listen = listen; |                         inbound4.listen = listen; | ||||||
|                         v2rayConfig.inbounds.Add(inbound4); |                         v2rayConfig.inbounds.Add(inbound4); | ||||||
| 
 | 
 | ||||||
|                         //auth |                         //auth | ||||||
|                         if (Utils.IsNotEmpty(_config.Inbound.First().User) && Utils.IsNotEmpty(_config.Inbound.First().Pass)) |                         if (Utils.IsNotEmpty(_config.Inbound[0].User) && Utils.IsNotEmpty(_config.Inbound[0].Pass)) | ||||||
|                         { |                         { | ||||||
|                             inbound3.settings.auth = "password"; |                             inbound3.settings.auth = "password"; | ||||||
|                             inbound3.settings.accounts = new List<AccountsItem4Ray> { new AccountsItem4Ray() { user = _config.Inbound.First().User, pass = _config.Inbound.First().Pass } }; |                             inbound3.settings.accounts = new List<AccountsItem4Ray> { new AccountsItem4Ray() { user = _config.Inbound[0].User, pass = _config.Inbound[0].Pass } }; | ||||||
| 
 | 
 | ||||||
|                             inbound4.settings.auth = "password"; |                             inbound4.settings.auth = "password"; | ||||||
|                             inbound4.settings.accounts = new List<AccountsItem4Ray> { new AccountsItem4Ray() { user = _config.Inbound.First().User, pass = _config.Inbound.First().Pass } }; |                             inbound4.settings.accounts = new List<AccountsItem4Ray> { new AccountsItem4Ray() { user = _config.Inbound[0].User, pass = _config.Inbound[0].Pass } }; | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                     else |                     else | ||||||
|  | @ -587,7 +587,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                             } |                             } | ||||||
|                             else |                             else | ||||||
|                             { |                             { | ||||||
|                                 vnextItem = outbound.settings.vnext.First(); |                                 vnextItem = outbound.settings.vnext[0]; | ||||||
|                             } |                             } | ||||||
|                             vnextItem.address = node.Address; |                             vnextItem.address = node.Address; | ||||||
|                             vnextItem.port = node.Port; |                             vnextItem.port = node.Port; | ||||||
|  | @ -600,7 +600,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                             } |                             } | ||||||
|                             else |                             else | ||||||
|                             { |                             { | ||||||
|                                 usersItem = vnextItem.users.First(); |                                 usersItem = vnextItem.users[0]; | ||||||
|                             } |                             } | ||||||
|                             //远程服务器用户ID |                             //远程服务器用户ID | ||||||
|                             usersItem.id = node.Id; |                             usersItem.id = node.Id; | ||||||
|  | @ -630,7 +630,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                             } |                             } | ||||||
|                             else |                             else | ||||||
|                             { |                             { | ||||||
|                                 serversItem = outbound.settings.servers.First(); |                                 serversItem = outbound.settings.servers[0]; | ||||||
|                             } |                             } | ||||||
|                             serversItem.address = node.Address; |                             serversItem.address = node.Address; | ||||||
|                             serversItem.port = node.Port; |                             serversItem.port = node.Port; | ||||||
|  | @ -656,7 +656,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                             } |                             } | ||||||
|                             else |                             else | ||||||
|                             { |                             { | ||||||
|                                 serversItem = outbound.settings.servers.First(); |                                 serversItem = outbound.settings.servers[0]; | ||||||
|                             } |                             } | ||||||
|                             serversItem.address = node.Address; |                             serversItem.address = node.Address; | ||||||
|                             serversItem.port = node.Port; |                             serversItem.port = node.Port; | ||||||
|  | @ -691,7 +691,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                             } |                             } | ||||||
|                             else |                             else | ||||||
|                             { |                             { | ||||||
|                                 vnextItem = outbound.settings.vnext.First(); |                                 vnextItem = outbound.settings.vnext[0]; | ||||||
|                             } |                             } | ||||||
|                             vnextItem.address = node.Address; |                             vnextItem.address = node.Address; | ||||||
|                             vnextItem.port = node.Port; |                             vnextItem.port = node.Port; | ||||||
|  | @ -704,7 +704,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                             } |                             } | ||||||
|                             else |                             else | ||||||
|                             { |                             { | ||||||
|                                 usersItem = vnextItem.users.First(); |                                 usersItem = vnextItem.users[0]; | ||||||
|                             } |                             } | ||||||
|                             usersItem.id = node.Id; |                             usersItem.id = node.Id; | ||||||
|                             usersItem.email = Global.UserEMail; |                             usersItem.email = Global.UserEMail; | ||||||
|  | @ -740,7 +740,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                             } |                             } | ||||||
|                             else |                             else | ||||||
|                             { |                             { | ||||||
|                                 serversItem = outbound.settings.servers.First(); |                                 serversItem = outbound.settings.servers[0]; | ||||||
|                             } |                             } | ||||||
|                             serversItem.address = node.Address; |                             serversItem.address = node.Address; | ||||||
|                             serversItem.port = node.Port; |                             serversItem.port = node.Port; | ||||||
|  | @ -1167,7 +1167,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|         { |         { | ||||||
|             //fragment proxy |             //fragment proxy | ||||||
|             if (_config.CoreBasicItem.EnableFragment |             if (_config.CoreBasicItem.EnableFragment | ||||||
|                 && Utils.IsNotEmpty(v2rayConfig.outbounds.First().streamSettings?.security)) |                 && Utils.IsNotEmpty(v2rayConfig.outbounds[0].streamSettings?.security)) | ||||||
|             { |             { | ||||||
|                 var fragmentOutbound = new Outbounds4Ray |                 var fragmentOutbound = new Outbounds4Ray | ||||||
|                 { |                 { | ||||||
|  | @ -1185,7 +1185,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                 }; |                 }; | ||||||
| 
 | 
 | ||||||
|                 v2rayConfig.outbounds.Add(fragmentOutbound); |                 v2rayConfig.outbounds.Add(fragmentOutbound); | ||||||
|                 v2rayConfig.outbounds.First().streamSettings.sockopt = new() |                 v2rayConfig.outbounds[0].streamSettings.sockopt = new() | ||||||
|                 { |                 { | ||||||
|                     dialerProxy = fragmentOutbound.tag |                     dialerProxy = fragmentOutbound.tag | ||||||
|                 }; |                 }; | ||||||
|  | @ -1205,7 +1205,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 //current proxy |                 //current proxy | ||||||
|                 var outbound = v2rayConfig.outbounds.First(); |                 var outbound = v2rayConfig.outbounds[0]; | ||||||
|                 var txtOutbound = Utils.GetEmbedText(Global.V2raySampleOutbound); |                 var txtOutbound = Utils.GetEmbedText(Global.V2raySampleOutbound); | ||||||
| 
 | 
 | ||||||
|                 //Previous proxy |                 //Previous proxy | ||||||
|  |  | ||||||
|  | @ -348,7 +348,7 @@ namespace ServiceLib.Services | ||||||
|                 if (!IPAddress.TryParse(url, out IPAddress? ipAddress)) |                 if (!IPAddress.TryParse(url, out IPAddress? ipAddress)) | ||||||
|                 { |                 { | ||||||
|                     IPHostEntry ipHostInfo = Dns.GetHostEntry(url); |                     IPHostEntry ipHostInfo = Dns.GetHostEntry(url); | ||||||
|                     ipAddress = ipHostInfo.AddressList.First(); |                     ipAddress = ipHostInfo.AddressList[0]; | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 var timer = Stopwatch.StartNew(); |                 var timer = Stopwatch.StartNew(); | ||||||
|  |  | ||||||
|  | @ -21,6 +21,9 @@ namespace ServiceLib.ViewModels | ||||||
|         [Reactive] |         [Reactive] | ||||||
|         public string HostFilter { get; set; } |         public string HostFilter { get; set; } | ||||||
| 
 | 
 | ||||||
|  |         [Reactive] | ||||||
|  |         public int SortingSelected { get; set; } | ||||||
|  | 
 | ||||||
|         [Reactive] |         [Reactive] | ||||||
|         public bool AutoRefresh { get; set; } |         public bool AutoRefresh { get; set; } | ||||||
| 
 | 
 | ||||||
|  | @ -28,12 +31,18 @@ namespace ServiceLib.ViewModels | ||||||
|         { |         { | ||||||
|             _config = AppHandler.Instance.Config; |             _config = AppHandler.Instance.Config; | ||||||
|             _updateView = updateView; |             _updateView = updateView; | ||||||
|  |             SortingSelected = _config.ClashUIItem.ConnectionsSorting; | ||||||
|             AutoRefresh = _config.ClashUIItem.ConnectionsAutoRefresh; |             AutoRefresh = _config.ClashUIItem.ConnectionsAutoRefresh; | ||||||
| 
 | 
 | ||||||
|             var canEditRemove = this.WhenAnyValue( |             var canEditRemove = this.WhenAnyValue( | ||||||
|              x => x.SelectedSource, |              x => x.SelectedSource, | ||||||
|              selectedSource => selectedSource != null && Utils.IsNotEmpty(selectedSource.Id)); |              selectedSource => selectedSource != null && Utils.IsNotEmpty(selectedSource.Id)); | ||||||
| 
 | 
 | ||||||
|  |             this.WhenAnyValue( | ||||||
|  |               x => x.SortingSelected, | ||||||
|  |               y => y >= 0) | ||||||
|  |                   .Subscribe(async c => await DoSortingSelected(c)); | ||||||
|  | 
 | ||||||
|             this.WhenAnyValue( |             this.WhenAnyValue( | ||||||
|                x => x.AutoRefresh, |                x => x.AutoRefresh, | ||||||
|                y => y == true) |                y => y == true) | ||||||
|  | @ -75,6 +84,20 @@ namespace ServiceLib.ViewModels | ||||||
|               }); |               }); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         private async Task DoSortingSelected(bool c) | ||||||
|  |         { | ||||||
|  |             if (!c) | ||||||
|  |             { | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |             if (SortingSelected != _config.ClashUIItem.ConnectionsSorting) | ||||||
|  |             { | ||||||
|  |                 _config.ClashUIItem.ConnectionsSorting = SortingSelected; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             await GetClashConnections(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         private async Task GetClashConnections() |         private async Task GetClashConnections() | ||||||
|         { |         { | ||||||
|             var ret = await ClashApiHandler.Instance.GetClashConnectionsAsync(_config); |             var ret = await ClashApiHandler.Instance.GetClashConnectionsAsync(_config); | ||||||
|  | @ -92,7 +115,7 @@ namespace ServiceLib.ViewModels | ||||||
| 
 | 
 | ||||||
|             var dtNow = DateTime.Now; |             var dtNow = DateTime.Now; | ||||||
|             var lstModel = new List<ClashConnectionModel>(); |             var lstModel = new List<ClashConnectionModel>(); | ||||||
|             foreach (var item in connections ?? new()) |             foreach (var item in connections ?? []) | ||||||
|             { |             { | ||||||
|                 var host = $"{(Utils.IsNullOrEmpty(item.metadata.host) ? item.metadata.destinationIP : item.metadata.host)}:{item.metadata.destinationPort}"; |                 var host = $"{(Utils.IsNullOrEmpty(item.metadata.host) ? item.metadata.destinationIP : item.metadata.host)}:{item.metadata.destinationPort}"; | ||||||
|                 if (HostFilter.IsNotEmpty() && !host.Contains(HostFilter)) |                 if (HostFilter.IsNotEmpty() && !host.Contains(HostFilter)) | ||||||
|  | @ -108,14 +131,45 @@ namespace ServiceLib.ViewModels | ||||||
|                 model.Host = host; |                 model.Host = host; | ||||||
|                 var sp = (dtNow - item.start); |                 var sp = (dtNow - item.start); | ||||||
|                 model.Time = sp.TotalSeconds < 0 ? 1 : sp.TotalSeconds; |                 model.Time = sp.TotalSeconds < 0 ? 1 : sp.TotalSeconds; | ||||||
|  |                 model.Upload = item.upload; | ||||||
|  |                 model.Download = item.download; | ||||||
|  |                 model.UploadTraffic = $"{Utils.HumanFy((long)item.upload)}"; | ||||||
|  |                 model.DownloadTraffic = $"{Utils.HumanFy((long)item.download)}"; | ||||||
|                 model.Elapsed = sp.ToString(@"hh\:mm\:ss"); |                 model.Elapsed = sp.ToString(@"hh\:mm\:ss"); | ||||||
|                 item.chains?.Reverse(); |                 model.Chain = item.chains?.Count > 0 ? item.chains[0] : string.Empty; | ||||||
|                 model.Chain = $"{item.rule} , {string.Join("->", item.chains ?? new())}"; |  | ||||||
| 
 | 
 | ||||||
|                 lstModel.Add(model); |                 lstModel.Add(model); | ||||||
|             } |             } | ||||||
|             if (lstModel.Count <= 0) { return; } |             if (lstModel.Count <= 0) { return; } | ||||||
| 
 | 
 | ||||||
|  |             //sort | ||||||
|  |             switch (SortingSelected) | ||||||
|  |             { | ||||||
|  |                 case 0: | ||||||
|  |                     lstModel = lstModel.OrderBy(t => t.Upload / t.Time).ToList(); | ||||||
|  |                     break; | ||||||
|  | 
 | ||||||
|  |                 case 1: | ||||||
|  |                     lstModel = lstModel.OrderBy(t => t.Download / t.Time).ToList(); | ||||||
|  |                     break; | ||||||
|  | 
 | ||||||
|  |                 case 2: | ||||||
|  |                     lstModel = lstModel.OrderBy(t => t.Upload).ToList(); | ||||||
|  |                     break; | ||||||
|  | 
 | ||||||
|  |                 case 3: | ||||||
|  |                     lstModel = lstModel.OrderBy(t => t.Download).ToList(); | ||||||
|  |                     break; | ||||||
|  | 
 | ||||||
|  |                 case 4: | ||||||
|  |                     lstModel = lstModel.OrderBy(t => t.Time).ToList(); | ||||||
|  |                     break; | ||||||
|  | 
 | ||||||
|  |                 case 5: | ||||||
|  |                     lstModel = lstModel.OrderBy(t => t.Host).ToList(); | ||||||
|  |                     break; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|             _connectionItems.AddRange(lstModel); |             _connectionItems.AddRange(lstModel); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -239,7 +239,7 @@ namespace ServiceLib.ViewModels | ||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|                 { |                 { | ||||||
|                     SelectedGroup = _proxyGroups.First(); |                     SelectedGroup = _proxyGroups[0]; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             else |             else | ||||||
|  |  | ||||||
|  | @ -122,7 +122,7 @@ namespace ServiceLib.ViewModels | ||||||
| 
 | 
 | ||||||
|             #region Core |             #region Core | ||||||
| 
 | 
 | ||||||
|             var inbound = _config.Inbound.First(); |             var inbound = _config.Inbound[0]; | ||||||
|             localPort = inbound.LocalPort; |             localPort = inbound.LocalPort; | ||||||
|             udpEnabled = inbound.UdpEnabled; |             udpEnabled = inbound.UdpEnabled; | ||||||
|             sniffingEnabled = inbound.SniffingEnabled; |             sniffingEnabled = inbound.SniffingEnabled; | ||||||
|  | @ -285,15 +285,15 @@ namespace ServiceLib.ViewModels | ||||||
|             //} |             //} | ||||||
| 
 | 
 | ||||||
|             //Core |             //Core | ||||||
|             _config.Inbound.First().LocalPort = localPort; |             _config.Inbound[0].LocalPort = localPort; | ||||||
|             _config.Inbound.First().UdpEnabled = udpEnabled; |             _config.Inbound[0].UdpEnabled = udpEnabled; | ||||||
|             _config.Inbound.First().SniffingEnabled = sniffingEnabled; |             _config.Inbound[0].SniffingEnabled = sniffingEnabled; | ||||||
|             _config.Inbound.First().DestOverride = destOverride?.ToList(); |             _config.Inbound[0].DestOverride = destOverride?.ToList(); | ||||||
|             _config.Inbound.First().RouteOnly = routeOnly; |             _config.Inbound[0].RouteOnly = routeOnly; | ||||||
|             _config.Inbound.First().AllowLANConn = allowLANConn; |             _config.Inbound[0].AllowLANConn = allowLANConn; | ||||||
|             _config.Inbound.First().NewPort4LAN = newPort4LAN; |             _config.Inbound[0].NewPort4LAN = newPort4LAN; | ||||||
|             _config.Inbound.First().User = user; |             _config.Inbound[0].User = user; | ||||||
|             _config.Inbound.First().Pass = pass; |             _config.Inbound[0].Pass = pass; | ||||||
|             if (_config.Inbound.Count > 1) |             if (_config.Inbound.Count > 1) | ||||||
|             { |             { | ||||||
|                 _config.Inbound.RemoveAt(1); |                 _config.Inbound.RemoveAt(1); | ||||||
|  |  | ||||||
|  | @ -374,7 +374,7 @@ namespace ServiceLib.ViewModels | ||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|                 { |                 { | ||||||
|                     SelectedProfile = lstModel.First(); |                     SelectedProfile = lstModel[0]; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  | @ -395,7 +395,7 @@ namespace ServiceLib.ViewModels | ||||||
|             } |             } | ||||||
|             else |             else | ||||||
|             { |             { | ||||||
|                 SelectedSub = _subItems.First(); |                 SelectedSub = _subItems[0]; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -456,9 +456,9 @@ namespace ServiceLib.ViewModels | ||||||
|             sb.Append($"[{EInboundProtocol.http}:{AppHandler.Instance.GetLocalPort(EInboundProtocol.http)}]"); |             sb.Append($"[{EInboundProtocol.http}:{AppHandler.Instance.GetLocalPort(EInboundProtocol.http)}]"); | ||||||
|             InboundDisplay = $"{ResUI.LabLocal}:{sb}"; |             InboundDisplay = $"{ResUI.LabLocal}:{sb}"; | ||||||
| 
 | 
 | ||||||
|             if (_config.Inbound.First().AllowLANConn) |             if (_config.Inbound[0].AllowLANConn) | ||||||
|             { |             { | ||||||
|                 if (_config.Inbound.First().NewPort4LAN) |                 if (_config.Inbound[0].NewPort4LAN) | ||||||
|                 { |                 { | ||||||
|                     StringBuilder sb2 = new(); |                     StringBuilder sb2 = new(); | ||||||
|                     sb2.Append($"[{EInboundProtocol.socks}:{AppHandler.Instance.GetLocalPort(EInboundProtocol.socks2)}]"); |                     sb2.Append($"[{EInboundProtocol.socks}:{AppHandler.Instance.GetLocalPort(EInboundProtocol.socks2)}]"); | ||||||
|  |  | ||||||
|  | @ -25,6 +25,22 @@ | ||||||
|                 VerticalContentAlignment="Center" |                 VerticalContentAlignment="Center" | ||||||
|                 Watermark="{x:Static resx:ResUI.ConnectionsHostFilterTitle}" /> |                 Watermark="{x:Static resx:ResUI.ConnectionsHostFilterTitle}" /> | ||||||
| 
 | 
 | ||||||
|  |             <TextBlock | ||||||
|  |                 Margin="8,0" | ||||||
|  |                 VerticalAlignment="Center" | ||||||
|  |                 Text="{x:Static resx:ResUI.TbSorting}" /> | ||||||
|  |             <ComboBox | ||||||
|  |                 x:Name="cmbSorting" | ||||||
|  |                 Width="100" | ||||||
|  |                 Margin="8,0"> | ||||||
|  |                 <ComboBoxItem Content="{x:Static resx:ResUI.TbSortingUpSpeed}" /> | ||||||
|  |                 <ComboBoxItem Content="{x:Static resx:ResUI.TbSortingDownSpeed}" /> | ||||||
|  |                 <ComboBoxItem Content="{x:Static resx:ResUI.TbSortingUpTraffic}" /> | ||||||
|  |                 <ComboBoxItem Content="{x:Static resx:ResUI.TbSortingDownTraffic}" /> | ||||||
|  |                 <ComboBoxItem Content="{x:Static resx:ResUI.TbSortingTime}" /> | ||||||
|  |                 <ComboBoxItem Content="{x:Static resx:ResUI.TbSortingHost}" /> | ||||||
|  |             </ComboBox> | ||||||
|  | 
 | ||||||
|             <Button |             <Button | ||||||
|                 x:Name="btnConnectionCloseAll" |                 x:Name="btnConnectionCloseAll" | ||||||
|                 Width="30" |                 Width="30" | ||||||
|  | @ -69,11 +85,11 @@ | ||||||
|             </DataGrid.ContextMenu> |             </DataGrid.ContextMenu> | ||||||
|             <DataGrid.Columns> |             <DataGrid.Columns> | ||||||
|                 <DataGridTextColumn |                 <DataGridTextColumn | ||||||
|                     Width="300" |                     Width="240" | ||||||
|                     Binding="{Binding Host}" |                     Binding="{Binding Host}" | ||||||
|                     Header="{x:Static resx:ResUI.TbSortingHost}" /> |                     Header="{x:Static resx:ResUI.TbSortingHost}" /> | ||||||
|                 <DataGridTextColumn |                 <DataGridTextColumn | ||||||
|                     Width="500" |                     Width="160" | ||||||
|                     Binding="{Binding Chain}" |                     Binding="{Binding Chain}" | ||||||
|                     Header="{x:Static resx:ResUI.TbSortingChain}" /> |                     Header="{x:Static resx:ResUI.TbSortingChain}" /> | ||||||
|                 <DataGridTextColumn |                 <DataGridTextColumn | ||||||
|  | @ -81,9 +97,17 @@ | ||||||
|                     Binding="{Binding Network}" |                     Binding="{Binding Network}" | ||||||
|                     Header="{x:Static resx:ResUI.TbSortingNetwork}" /> |                     Header="{x:Static resx:ResUI.TbSortingNetwork}" /> | ||||||
|                 <DataGridTextColumn |                 <DataGridTextColumn | ||||||
|                     Width="160" |                     Width="100" | ||||||
|                     Binding="{Binding Type}" |                     Binding="{Binding Type}" | ||||||
|                     Header="{x:Static resx:ResUI.TbSortingType}" /> |                     Header="{x:Static resx:ResUI.TbSortingType}" /> | ||||||
|  |                 <DataGridTextColumn | ||||||
|  |                     Width="100" | ||||||
|  |                     Binding="{Binding UploadTraffic}" | ||||||
|  |                     Header="{x:Static resx:ResUI.TbSortingUpTraffic}" /> | ||||||
|  |                 <DataGridTextColumn | ||||||
|  |                     Width="100" | ||||||
|  |                     Binding="{Binding DownloadTraffic}" | ||||||
|  |                     Header="{x:Static resx:ResUI.TbSortingDownTraffic}" /> | ||||||
|                 <DataGridTextColumn |                 <DataGridTextColumn | ||||||
|                     Width="100" |                     Width="100" | ||||||
|                     Binding="{Binding Elapsed}" |                     Binding="{Binding Elapsed}" | ||||||
|  |  | ||||||
|  | @ -22,6 +22,7 @@ namespace v2rayN.Desktop.Views | ||||||
|                 this.BindCommand(ViewModel, vm => vm.ConnectionCloseAllCmd, v => v.menuConnectionCloseAll).DisposeWith(disposables); |                 this.BindCommand(ViewModel, vm => vm.ConnectionCloseAllCmd, v => v.menuConnectionCloseAll).DisposeWith(disposables); | ||||||
| 
 | 
 | ||||||
|                 this.Bind(ViewModel, vm => vm.HostFilter, v => v.txtHostFilter.Text).DisposeWith(disposables); |                 this.Bind(ViewModel, vm => vm.HostFilter, v => v.txtHostFilter.Text).DisposeWith(disposables); | ||||||
|  |                 this.Bind(ViewModel, vm => vm.SortingSelected, v => v.cmbSorting.SelectedIndex).DisposeWith(disposables); | ||||||
|                 this.BindCommand(ViewModel, vm => vm.ConnectionCloseAllCmd, v => v.btnConnectionCloseAll).DisposeWith(disposables); |                 this.BindCommand(ViewModel, vm => vm.ConnectionCloseAllCmd, v => v.btnConnectionCloseAll).DisposeWith(disposables); | ||||||
|                 this.Bind(ViewModel, vm => vm.AutoRefresh, v => v.togAutoRefresh.IsChecked).DisposeWith(disposables); |                 this.Bind(ViewModel, vm => vm.AutoRefresh, v => v.togAutoRefresh.IsChecked).DisposeWith(disposables); | ||||||
|             }); |             }); | ||||||
|  |  | ||||||
|  | @ -23,7 +23,7 @@ namespace v2rayN.Desktop.Views | ||||||
|             { |             { | ||||||
|                 clbdestOverride.Items.Add(it); |                 clbdestOverride.Items.Add(it); | ||||||
|             }); |             }); | ||||||
|             _config.Inbound.First().DestOverride?.ForEach(it => |             _config.Inbound[0].DestOverride?.ForEach(it => | ||||||
|             { |             { | ||||||
|                 clbdestOverride.SelectedItems.Add(it); |                 clbdestOverride.SelectedItems.Add(it); | ||||||
|             }); |             }); | ||||||
|  |  | ||||||
|  | @ -1,11 +1,11 @@ | ||||||
| <reactiveui:ReactiveUserControl | <reactiveui:ReactiveUserControl | ||||||
|     x:Class="v2rayN.Views.ClashConnectionsView" |     x:Class="v2rayN.Views.ClashConnectionsView" | ||||||
|     xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" |     xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" | ||||||
|     xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" |  | ||||||
|     xmlns:d="http://schemas.microsoft.com/expression/blend/2008" |  | ||||||
|     xmlns:materialDesign="http://materialdesigninxaml.net/winfx/xaml/themes" |     xmlns:materialDesign="http://materialdesigninxaml.net/winfx/xaml/themes" | ||||||
|     xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" |  | ||||||
|     xmlns:reactiveui="http://reactiveui.net" |     xmlns:reactiveui="http://reactiveui.net" | ||||||
|  |     xmlns:d="http://schemas.microsoft.com/expression/blend/2008" | ||||||
|  |     xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" | ||||||
|  |     xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" | ||||||
|     xmlns:resx="clr-namespace:ServiceLib.Resx;assembly=ServiceLib" |     xmlns:resx="clr-namespace:ServiceLib.Resx;assembly=ServiceLib" | ||||||
|     xmlns:vms="clr-namespace:ServiceLib.ViewModels;assembly=ServiceLib" |     xmlns:vms="clr-namespace:ServiceLib.ViewModels;assembly=ServiceLib" | ||||||
|     d:DesignHeight="450" |     d:DesignHeight="450" | ||||||
|  | @ -29,6 +29,24 @@ | ||||||
|                 materialDesign:TextFieldAssist.HasClearButton="True" |                 materialDesign:TextFieldAssist.HasClearButton="True" | ||||||
|                 Style="{StaticResource DefTextBox}" /> |                 Style="{StaticResource DefTextBox}" /> | ||||||
| 
 | 
 | ||||||
|  |             <TextBlock | ||||||
|  |                 Margin="{StaticResource MarginLeftRight8}" | ||||||
|  |                 VerticalAlignment="Center" | ||||||
|  |                 Style="{StaticResource ToolbarTextBlock}" | ||||||
|  |                 Text="{x:Static resx:ResUI.TbSorting}" /> | ||||||
|  |             <ComboBox | ||||||
|  |                 x:Name="cmbSorting" | ||||||
|  |                 Width="100" | ||||||
|  |                 Margin="{StaticResource MarginLeftRight8}" | ||||||
|  |                 Style="{StaticResource DefComboBox}"> | ||||||
|  |                 <ComboBoxItem Content="{x:Static resx:ResUI.TbSortingUpSpeed}" /> | ||||||
|  |                 <ComboBoxItem Content="{x:Static resx:ResUI.TbSortingDownSpeed}" /> | ||||||
|  |                 <ComboBoxItem Content="{x:Static resx:ResUI.TbSortingUpTraffic}" /> | ||||||
|  |                 <ComboBoxItem Content="{x:Static resx:ResUI.TbSortingDownTraffic}" /> | ||||||
|  |                 <ComboBoxItem Content="{x:Static resx:ResUI.TbSortingTime}" /> | ||||||
|  |                 <ComboBoxItem Content="{x:Static resx:ResUI.TbSortingHost}" /> | ||||||
|  |             </ComboBox> | ||||||
|  | 
 | ||||||
|             <Button |             <Button | ||||||
|                 x:Name="btnConnectionCloseAll" |                 x:Name="btnConnectionCloseAll" | ||||||
|                 Width="24" |                 Width="24" | ||||||
|  | @ -70,11 +88,11 @@ | ||||||
|             </DataGrid.ContextMenu> |             </DataGrid.ContextMenu> | ||||||
|             <DataGrid.Columns> |             <DataGrid.Columns> | ||||||
|                 <DataGridTextColumn |                 <DataGridTextColumn | ||||||
|                     Width="300" |                     Width="240" | ||||||
|                     Binding="{Binding Host}" |                     Binding="{Binding Host}" | ||||||
|                     Header="{x:Static resx:ResUI.TbSortingHost}" /> |                     Header="{x:Static resx:ResUI.TbSortingHost}" /> | ||||||
|                 <DataGridTextColumn |                 <DataGridTextColumn | ||||||
|                     Width="500" |                     Width="160" | ||||||
|                     Binding="{Binding Chain}" |                     Binding="{Binding Chain}" | ||||||
|                     Header="{x:Static resx:ResUI.TbSortingChain}" /> |                     Header="{x:Static resx:ResUI.TbSortingChain}" /> | ||||||
|                 <DataGridTextColumn |                 <DataGridTextColumn | ||||||
|  | @ -82,9 +100,17 @@ | ||||||
|                     Binding="{Binding Network}" |                     Binding="{Binding Network}" | ||||||
|                     Header="{x:Static resx:ResUI.TbSortingNetwork}" /> |                     Header="{x:Static resx:ResUI.TbSortingNetwork}" /> | ||||||
|                 <DataGridTextColumn |                 <DataGridTextColumn | ||||||
|                     Width="160" |                     Width="100" | ||||||
|                     Binding="{Binding Type}" |                     Binding="{Binding Type}" | ||||||
|                     Header="{x:Static resx:ResUI.TbSortingType}" /> |                     Header="{x:Static resx:ResUI.TbSortingType}" /> | ||||||
|  |                 <DataGridTextColumn | ||||||
|  |                     Width="100" | ||||||
|  |                     Binding="{Binding UploadTraffic}" | ||||||
|  |                     Header="{x:Static resx:ResUI.TbSortingUpTraffic}" /> | ||||||
|  |                 <DataGridTextColumn | ||||||
|  |                     Width="100" | ||||||
|  |                     Binding="{Binding DownloadTraffic}" | ||||||
|  |                     Header="{x:Static resx:ResUI.TbSortingDownTraffic}" /> | ||||||
|                 <DataGridTextColumn |                 <DataGridTextColumn | ||||||
|                     Width="100" |                     Width="100" | ||||||
|                     Binding="{Binding Elapsed}" |                     Binding="{Binding Elapsed}" | ||||||
|  |  | ||||||
|  | @ -24,6 +24,7 @@ namespace v2rayN.Views | ||||||
|                 this.BindCommand(ViewModel, vm => vm.ConnectionCloseAllCmd, v => v.menuConnectionCloseAll).DisposeWith(disposables); |                 this.BindCommand(ViewModel, vm => vm.ConnectionCloseAllCmd, v => v.menuConnectionCloseAll).DisposeWith(disposables); | ||||||
| 
 | 
 | ||||||
|                 this.Bind(ViewModel, vm => vm.HostFilter, v => v.txtHostFilter.Text).DisposeWith(disposables); |                 this.Bind(ViewModel, vm => vm.HostFilter, v => v.txtHostFilter.Text).DisposeWith(disposables); | ||||||
|  |                 this.Bind(ViewModel, vm => vm.SortingSelected, v => v.cmbSorting.SelectedIndex).DisposeWith(disposables); | ||||||
|                 this.BindCommand(ViewModel, vm => vm.ConnectionCloseAllCmd, v => v.btnConnectionCloseAll).DisposeWith(disposables); |                 this.BindCommand(ViewModel, vm => vm.ConnectionCloseAllCmd, v => v.btnConnectionCloseAll).DisposeWith(disposables); | ||||||
|                 this.Bind(ViewModel, vm => vm.AutoRefresh, v => v.togAutoRefresh.IsChecked).DisposeWith(disposables); |                 this.Bind(ViewModel, vm => vm.AutoRefresh, v => v.togAutoRefresh.IsChecked).DisposeWith(disposables); | ||||||
|             }); |             }); | ||||||
|  |  | ||||||
|  | @ -25,7 +25,7 @@ namespace v2rayN.Views | ||||||
|             { |             { | ||||||
|                 clbdestOverride.Items.Add(it); |                 clbdestOverride.Items.Add(it); | ||||||
|             }); |             }); | ||||||
|             _config.Inbound.First().DestOverride?.ForEach(it => |             _config.Inbound[0].DestOverride?.ForEach(it => | ||||||
|             { |             { | ||||||
|                 clbdestOverride.SelectedItems.Add(it); |                 clbdestOverride.SelectedItems.Add(it); | ||||||
|             }); |             }); | ||||||
|  | @ -211,7 +211,7 @@ namespace v2rayN.Views | ||||||
|                 { |                 { | ||||||
|                     files.AddRange(Directory.GetFiles(path, pattern)); |                     files.AddRange(Directory.GetFiles(path, pattern)); | ||||||
|                 } |                 } | ||||||
|                 var culture = _config.UiItem.CurrentLanguage == Global.Languages.First() ? "zh-cn" : "en-us"; |                 var culture = _config.UiItem.CurrentLanguage == Global.Languages[0] ? "zh-cn" : "en-us"; | ||||||
|                 var culture2 = "en-us"; |                 var culture2 = "en-us"; | ||||||
|                 foreach (var ttf in files) |                 foreach (var ttf in files) | ||||||
|                 { |                 { | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue