mirror of
				https://github.com/2dust/v2rayN.git
				synced 2025-10-26 02:04:40 +00:00 
			
		
		
		
	Compare commits
	
		
			27 commits
		
	
	
		
			84ca5808b9
			...
			ea7ea5a0bf
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | ea7ea5a0bf | ||
|   | a652fd879b | ||
|   | 326bf334e7 | ||
|   | 21a773f400 | ||
|   | c88bf796f4 | ||
|   | ab1dc45ed4 | ||
|   | 2d41272659 | ||
|   | e7f75010d3 | ||
|   | aa1ccdd01b | ||
|   | b17323c982 | ||
|   | 71dcd8d1de | ||
|   | dace865e6c | ||
|   | fad94e68d2 | ||
|   | c116aae242 | ||
|   | ff1c9093a2 | ||
|   | 8bb20c0ab8 | ||
|   | 17a3a516c7 | ||
|   | 8f0d7e54d8 | ||
|   | 2ab79afa02 | ||
|   | e29d292732 | ||
|   | 9ef228db1e | ||
|   | 34327532e6 | ||
|   | 8af6eda165 | ||
|   | f979d13109 | ||
|   | 6166b6c0e3 | ||
|   | 8c094dd976 | ||
|   | 5c4f485471 | 
					 54 changed files with 2828 additions and 486 deletions
				
			
		
							
								
								
									
										2
									
								
								.github/workflows/build-linux.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/build-linux.yml
									
									
									
									
										vendored
									
									
								
							|  | @ -22,7 +22,7 @@ jobs: | ||||||
|       matrix: |       matrix: | ||||||
|         configuration: [Release] |         configuration: [Release] | ||||||
| 
 | 
 | ||||||
|     runs-on: ubuntu-22.04 |     runs-on: ubuntu-24.04 | ||||||
| 
 | 
 | ||||||
|     steps: |     steps: | ||||||
|     - name: Checkout |     - name: Checkout | ||||||
|  |  | ||||||
|  | @ -5,6 +5,7 @@ | ||||||
|     <CentralPackageVersionOverrideEnabled>false</CentralPackageVersionOverrideEnabled> |     <CentralPackageVersionOverrideEnabled>false</CentralPackageVersionOverrideEnabled> | ||||||
|   </PropertyGroup> |   </PropertyGroup> | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|  |     <PackageVersion Include="Avalonia.AvaloniaEdit" Version="11.3.0" /> | ||||||
|     <PackageVersion Include="Avalonia.Controls.DataGrid" Version="11.3.6" /> |     <PackageVersion Include="Avalonia.Controls.DataGrid" Version="11.3.6" /> | ||||||
|     <PackageVersion Include="Avalonia.Desktop" Version="11.3.6" /> |     <PackageVersion Include="Avalonia.Desktop" Version="11.3.6" /> | ||||||
|     <PackageVersion Include="Avalonia.Diagnostics" Version="11.3.6" /> |     <PackageVersion Include="Avalonia.Diagnostics" Version="11.3.6" /> | ||||||
|  | @ -19,6 +20,7 @@ | ||||||
|     <PackageVersion Include="ReactiveUI.Fody" Version="19.5.41" /> |     <PackageVersion Include="ReactiveUI.Fody" Version="19.5.41" /> | ||||||
|     <PackageVersion Include="ReactiveUI.WPF" Version="20.4.1" /> |     <PackageVersion Include="ReactiveUI.WPF" Version="20.4.1" /> | ||||||
|     <PackageVersion Include="Semi.Avalonia" Version="11.2.1.10" /> |     <PackageVersion Include="Semi.Avalonia" Version="11.2.1.10" /> | ||||||
|  |     <PackageVersion Include="Semi.Avalonia.AvaloniaEdit" Version="11.2.0.1" /> | ||||||
|     <PackageVersion Include="Semi.Avalonia.DataGrid" Version="11.2.1.10" /> |     <PackageVersion Include="Semi.Avalonia.DataGrid" Version="11.2.1.10" /> | ||||||
|     <PackageVersion Include="NLog" Version="6.0.4" /> |     <PackageVersion Include="NLog" Version="6.0.4" /> | ||||||
|     <PackageVersion Include="sqlite-net-pcl" Version="1.9.172" /> |     <PackageVersion Include="sqlite-net-pcl" Version="1.9.172" /> | ||||||
|  |  | ||||||
|  | @ -12,5 +12,9 @@ public enum EConfigType | ||||||
|     TUIC = 8, |     TUIC = 8, | ||||||
|     WireGuard = 9, |     WireGuard = 9, | ||||||
|     HTTP = 10, |     HTTP = 10, | ||||||
|     Anytls = 11 |     Anytls = 11, | ||||||
|  | 
 | ||||||
|  |     Group = 1000, | ||||||
|  |     PolicyGroup = 1001, | ||||||
|  |     ProxyChain = 1002, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -2,8 +2,9 @@ namespace ServiceLib.Enums; | ||||||
| 
 | 
 | ||||||
| public enum EMultipleLoad | public enum EMultipleLoad | ||||||
| { | { | ||||||
|  |     LeastPing, | ||||||
|  |     Fallback, | ||||||
|     Random, |     Random, | ||||||
|     RoundRobin, |     RoundRobin, | ||||||
|     LeastPing, |  | ||||||
|     LeastLoad |     LeastLoad | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -23,6 +23,7 @@ public enum EViewAction | ||||||
|     RoutingRuleDetailsWindow, |     RoutingRuleDetailsWindow, | ||||||
|     AddServerWindow, |     AddServerWindow, | ||||||
|     AddServer2Window, |     AddServer2Window, | ||||||
|  |     AddGroupServerWindow, | ||||||
|     DNSSettingWindow, |     DNSSettingWindow, | ||||||
|     RoutingSettingWindow, |     RoutingSettingWindow, | ||||||
|     OptionSettingWindow, |     OptionSettingWindow, | ||||||
|  |  | ||||||
|  | @ -50,6 +50,7 @@ public class Global | ||||||
|     public const string DirectTag = "direct"; |     public const string DirectTag = "direct"; | ||||||
|     public const string BlockTag = "block"; |     public const string BlockTag = "block"; | ||||||
|     public const string DnsTag = "dns-module"; |     public const string DnsTag = "dns-module"; | ||||||
|  |     public const string BalancerTagSuffix = "-round"; | ||||||
|     public const string StreamSecurity = "tls"; |     public const string StreamSecurity = "tls"; | ||||||
|     public const string StreamSecurityReality = "reality"; |     public const string StreamSecurityReality = "reality"; | ||||||
|     public const string Loopback = "127.0.0.1"; |     public const string Loopback = "127.0.0.1"; | ||||||
|  | @ -449,6 +450,14 @@ public class Global | ||||||
|             "none" |             "none" | ||||||
|     ]; |     ]; | ||||||
| 
 | 
 | ||||||
|  |     public static readonly Dictionary<string, string> LogLevelColors = new() | ||||||
|  |     { | ||||||
|  |         { "debug",   "#6C757D" }, | ||||||
|  |         { "info",    "#2ECC71" }, | ||||||
|  |         { "warning", "#FFA500" }, | ||||||
|  |         { "error",   "#E74C3C" }, | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|     public static readonly List<string> InboundTags = |     public static readonly List<string> InboundTags = | ||||||
|     [ |     [ | ||||||
|         "socks", |         "socks", | ||||||
|  |  | ||||||
|  | @ -357,6 +357,11 @@ public static class ConfigHandler | ||||||
|                 { |                 { | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |             else if (profileItem.ConfigType > EConfigType.Group) | ||||||
|  |             { | ||||||
|  |                 var profileGroupItem = await AppManager.Instance.GetProfileGroupItem(it.IndexId); | ||||||
|  |                 await AddGroupServerCommon(config, profileItem, profileGroupItem, true); | ||||||
|  |             } | ||||||
|             else |             else | ||||||
|             { |             { | ||||||
|                 await AddServerCommon(config, profileItem, true); |                 await AddServerCommon(config, profileItem, true); | ||||||
|  | @ -1074,6 +1079,35 @@ public static class ConfigHandler | ||||||
|         return 0; |         return 0; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     public static async Task<int> AddGroupServerCommon(Config config, ProfileItem profileItem, ProfileGroupItem profileGroupItem, bool toFile = true) | ||||||
|  |     { | ||||||
|  |         var maxSort = -1; | ||||||
|  |         if (profileItem.IndexId.IsNullOrEmpty()) | ||||||
|  |         { | ||||||
|  |             profileItem.IndexId = Utils.GetGuid(false); | ||||||
|  |             maxSort = ProfileExManager.Instance.GetMaxSort(); | ||||||
|  |         } | ||||||
|  |         if (maxSort > 0) | ||||||
|  |         { | ||||||
|  |             ProfileExManager.Instance.SetSort(profileItem.IndexId, maxSort + 1); | ||||||
|  |         } | ||||||
|  |         if (toFile) | ||||||
|  |         { | ||||||
|  |             await SQLiteHelper.Instance.ReplaceAsync(profileItem); | ||||||
|  |             if (profileGroupItem != null) | ||||||
|  |             { | ||||||
|  |                 profileGroupItem.ParentIndexId = profileItem.IndexId; | ||||||
|  |                 await ProfileGroupItemManager.Instance.SaveItemAsync(profileGroupItem); | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 ProfileGroupItemManager.Instance.GetOrCreateAndMarkDirty(profileItem.IndexId); | ||||||
|  |                 await ProfileGroupItemManager.Instance.SaveTo(); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return 0; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Compare two profile items to determine if they represent the same server |     /// Compare two profile items to determine if they represent the same server | ||||||
|     /// Used for deduplication and server matching |     /// Used for deduplication and server matching | ||||||
|  | @ -1145,7 +1179,7 @@ public static class ConfigHandler | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /// <summary> |     /// <summary> | ||||||
|     /// Create a custom server that combines multiple servers for load balancing |     /// Create a group server that combines multiple servers for load balancing | ||||||
|     /// Generates a configuration file that references multiple servers |     /// Generates a configuration file that references multiple servers | ||||||
|     /// </summary> |     /// </summary> | ||||||
|     /// <param name="config">Current configuration</param> |     /// <param name="config">Current configuration</param> | ||||||
|  | @ -1153,45 +1187,55 @@ public static class ConfigHandler | ||||||
|     /// <param name="coreType">Core type to use (Xray or sing_box)</param> |     /// <param name="coreType">Core type to use (Xray or sing_box)</param> | ||||||
|     /// <param name="multipleLoad">Load balancing algorithm</param> |     /// <param name="multipleLoad">Load balancing algorithm</param> | ||||||
|     /// <returns>Result object with success state and data</returns> |     /// <returns>Result object with success state and data</returns> | ||||||
|     public static async Task<RetResult> AddCustomServer4Multiple(Config config, List<ProfileItem> selecteds, ECoreType coreType, EMultipleLoad multipleLoad) |     public static async Task<RetResult> AddGroupServer4Multiple(Config config, List<ProfileItem> selecteds, ECoreType coreType, EMultipleLoad multipleLoad, string? subId) | ||||||
|     { |     { | ||||||
|         var indexId = Utils.GetMd5(Global.CoreMultipleLoadConfigFileName); |         var result = new RetResult(); | ||||||
|         var configPath = Utils.GetConfigPath(Global.CoreMultipleLoadConfigFileName); |  | ||||||
| 
 | 
 | ||||||
|         var result = await CoreConfigHandler.GenerateClientMultipleLoadConfig(config, configPath, selecteds, coreType, multipleLoad); |         var indexId = Utils.GetGuid(false); | ||||||
|         if (result.Success != true) |         var childProfileIndexId = Utils.List2String(selecteds.Select(p => p.IndexId).ToList()); | ||||||
|         { |  | ||||||
|             return result; |  | ||||||
|         } |  | ||||||
| 
 | 
 | ||||||
|         if (!File.Exists(configPath)) |         var remark = string.Empty; | ||||||
|         { |  | ||||||
|             return result; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         var profileItem = await AppManager.Instance.GetProfileItem(indexId) ?? new(); |  | ||||||
|         profileItem.IndexId = indexId; |  | ||||||
|         if (coreType == ECoreType.Xray) |         if (coreType == ECoreType.Xray) | ||||||
|         { |         { | ||||||
|             profileItem.Remarks = multipleLoad switch |             remark = multipleLoad switch | ||||||
|             { |             { | ||||||
|                 EMultipleLoad.Random => ResUI.menuSetDefaultMultipleServerXrayRandom, |                 EMultipleLoad.LeastPing => ResUI.menuGenGroupMultipleServerXrayLeastPing, | ||||||
|                 EMultipleLoad.RoundRobin => ResUI.menuSetDefaultMultipleServerXrayRoundRobin, |                 EMultipleLoad.Fallback => ResUI.menuGenGroupMultipleServerXrayFallback, | ||||||
|                 EMultipleLoad.LeastPing => ResUI.menuSetDefaultMultipleServerXrayLeastPing, |                 EMultipleLoad.Random => ResUI.menuGenGroupMultipleServerXrayRandom, | ||||||
|                 EMultipleLoad.LeastLoad => ResUI.menuSetDefaultMultipleServerXrayLeastLoad, |                 EMultipleLoad.RoundRobin => ResUI.menuGenGroupMultipleServerXrayRoundRobin, | ||||||
|                 _ => ResUI.menuSetDefaultMultipleServerXrayRoundRobin, |                 EMultipleLoad.LeastLoad => ResUI.menuGenGroupMultipleServerXrayLeastLoad, | ||||||
|  |                 _ => ResUI.menuGenGroupMultipleServerXrayRoundRobin, | ||||||
|             }; |             }; | ||||||
|         } |         } | ||||||
|         else if (coreType == ECoreType.sing_box) |         else if (coreType == ECoreType.sing_box) | ||||||
|         { |         { | ||||||
|             profileItem.Remarks = ResUI.menuSetDefaultMultipleServerSingBoxLeastPing; |             remark = multipleLoad switch | ||||||
|  |             { | ||||||
|  |                 EMultipleLoad.LeastPing => ResUI.menuGenGroupMultipleServerSingBoxLeastPing, | ||||||
|  |                 EMultipleLoad.Fallback => ResUI.menuGenGroupMultipleServerSingBoxFallback, | ||||||
|  |                 _ => ResUI.menuGenGroupMultipleServerSingBoxLeastPing, | ||||||
|  |             }; | ||||||
|         } |         } | ||||||
|         profileItem.Address = Global.CoreMultipleLoadConfigFileName; |         var profile = new ProfileItem | ||||||
|         profileItem.ConfigType = EConfigType.Custom; |         { | ||||||
|         profileItem.CoreType = coreType; |             IndexId = indexId, | ||||||
| 
 |             CoreType = coreType, | ||||||
|         await AddServerCommon(config, profileItem, true); |             ConfigType = EConfigType.PolicyGroup, | ||||||
| 
 |             Remarks = remark, | ||||||
|  |             Address = childProfileIndexId, | ||||||
|  |         }; | ||||||
|  |         if (!subId.IsNullOrEmpty()) | ||||||
|  |         { | ||||||
|  |             profile.Subid = subId; | ||||||
|  |         } | ||||||
|  |         var profileGroup = new ProfileGroupItem | ||||||
|  |         { | ||||||
|  |             ChildItems = childProfileIndexId, | ||||||
|  |             MultipleLoad = multipleLoad, | ||||||
|  |             ParentIndexId = indexId, | ||||||
|  |         }; | ||||||
|  |         var ret = await AddGroupServerCommon(config, profile, profileGroup, true); | ||||||
|  |         result.Success = ret == 0; | ||||||
|         result.Data = indexId; |         result.Data = indexId; | ||||||
|         return result; |         return result; | ||||||
|     } |     } | ||||||
|  | @ -1209,12 +1253,49 @@ public static class ConfigHandler | ||||||
|         ProfileItem? itemSocks = null; |         ProfileItem? itemSocks = null; | ||||||
|         if (node.ConfigType != EConfigType.Custom && coreType != ECoreType.sing_box && config.TunModeItem.EnableTun) |         if (node.ConfigType != EConfigType.Custom && coreType != ECoreType.sing_box && config.TunModeItem.EnableTun) | ||||||
|         { |         { | ||||||
|  |             var tun2SocksAddress = node.Address; | ||||||
|  |             if (node.ConfigType > EConfigType.Group) | ||||||
|  |             { | ||||||
|  |                 static async Task<List<string>> GetChildNodeAddressesAsync(string parentIndexId) | ||||||
|  |                 { | ||||||
|  |                     var childAddresses = new List<string>(); | ||||||
|  |                     if (!ProfileGroupItemManager.Instance.TryGet(parentIndexId, out var groupItem) || groupItem.ChildItems.IsNullOrEmpty()) | ||||||
|  |                         return childAddresses; | ||||||
|  | 
 | ||||||
|  |                     var childIds = Utils.String2List(groupItem.ChildItems); | ||||||
|  | 
 | ||||||
|  |                     foreach (var childId in childIds) | ||||||
|  |                     { | ||||||
|  |                         var childNode = await AppManager.Instance.GetProfileItem(childId); | ||||||
|  |                         if (childNode == null) | ||||||
|  |                             continue; | ||||||
|  | 
 | ||||||
|  |                         if (!childNode.IsComplex()) | ||||||
|  |                         { | ||||||
|  |                             childAddresses.Add(childNode.Address); | ||||||
|  |                         } | ||||||
|  |                         else if (childNode.ConfigType > EConfigType.Group) | ||||||
|  |                         { | ||||||
|  |                             var subAddresses = await GetChildNodeAddressesAsync(childNode.IndexId); | ||||||
|  |                             childAddresses.AddRange(subAddresses); | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     return childAddresses; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 var lstAddresses = await GetChildNodeAddressesAsync(node.IndexId); | ||||||
|  |                 if (lstAddresses.Count > 0) | ||||||
|  |                 { | ||||||
|  |                     tun2SocksAddress = Utils.List2String(lstAddresses); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|             itemSocks = new ProfileItem() |             itemSocks = new ProfileItem() | ||||||
|             { |             { | ||||||
|                 CoreType = ECoreType.sing_box, |                 CoreType = ECoreType.sing_box, | ||||||
|                 ConfigType = EConfigType.SOCKS, |                 ConfigType = EConfigType.SOCKS, | ||||||
|                 Address = Global.Loopback, |                 Address = Global.Loopback, | ||||||
|                 SpiderX = node.Address, // Tun2SocksAddress |                 SpiderX = tun2SocksAddress, // Tun2SocksAddress | ||||||
|                 Port = AppManager.Instance.GetLocalPort(EInboundProtocol.socks) |                 Port = AppManager.Instance.GetLocalPort(EInboundProtocol.socks) | ||||||
|             }; |             }; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  | @ -132,24 +132,4 @@ public static class CoreConfigHandler | ||||||
|         await File.WriteAllTextAsync(fileName, result.Data.ToString()); |         await File.WriteAllTextAsync(fileName, result.Data.ToString()); | ||||||
|         return result; |         return result; | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     public static async Task<RetResult> GenerateClientMultipleLoadConfig(Config config, string fileName, List<ProfileItem> selecteds, ECoreType coreType, EMultipleLoad multipleLoad) |  | ||||||
|     { |  | ||||||
|         var result = new RetResult(); |  | ||||||
|         if (coreType == ECoreType.sing_box) |  | ||||||
|         { |  | ||||||
|             result = await new CoreConfigSingboxService(config).GenerateClientMultipleLoadConfig(selecteds); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             result = await new CoreConfigV2rayService(config).GenerateClientMultipleLoadConfig(selecteds, multipleLoad); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if (result.Success != true) |  | ||||||
|         { |  | ||||||
|             return result; |  | ||||||
|         } |  | ||||||
|         await File.WriteAllTextAsync(fileName, result.Data.ToString()); |  | ||||||
|         return result; |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -67,6 +67,7 @@ public sealed class AppManager | ||||||
|         SQLiteHelper.Instance.CreateTable<ProfileExItem>(); |         SQLiteHelper.Instance.CreateTable<ProfileExItem>(); | ||||||
|         SQLiteHelper.Instance.CreateTable<DNSItem>(); |         SQLiteHelper.Instance.CreateTable<DNSItem>(); | ||||||
|         SQLiteHelper.Instance.CreateTable<FullConfigTemplateItem>(); |         SQLiteHelper.Instance.CreateTable<FullConfigTemplateItem>(); | ||||||
|  |         SQLiteHelper.Instance.CreateTable<ProfileGroupItem>(); | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -101,6 +102,7 @@ public sealed class AppManager | ||||||
| 
 | 
 | ||||||
|             await ConfigHandler.SaveConfig(_config); |             await ConfigHandler.SaveConfig(_config); | ||||||
|             await ProfileExManager.Instance.SaveTo(); |             await ProfileExManager.Instance.SaveTo(); | ||||||
|  |             await ProfileGroupItemManager.Instance.SaveTo(); | ||||||
|             await StatisticsManager.Instance.SaveTo(); |             await StatisticsManager.Instance.SaveTo(); | ||||||
|             await CoreManager.Instance.CoreStop(); |             await CoreManager.Instance.CoreStop(); | ||||||
|             StatisticsManager.Instance.Close(); |             StatisticsManager.Instance.Close(); | ||||||
|  | @ -225,6 +227,15 @@ public sealed class AppManager | ||||||
|         return await SQLiteHelper.Instance.TableAsync<ProfileItem>().FirstOrDefaultAsync(it => it.Remarks == remarks); |         return await SQLiteHelper.Instance.TableAsync<ProfileItem>().FirstOrDefaultAsync(it => it.Remarks == remarks); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     public async Task<ProfileGroupItem?> GetProfileGroupItem(string parentIndexId) | ||||||
|  |     { | ||||||
|  |         if (parentIndexId.IsNullOrEmpty()) | ||||||
|  |         { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         return await SQLiteHelper.Instance.TableAsync<ProfileGroupItem>().FirstOrDefaultAsync(it => it.ParentIndexId == parentIndexId); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     public async Task<List<RoutingItem>?> RoutingItems() |     public async Task<List<RoutingItem>?> RoutingItems() | ||||||
|     { |     { | ||||||
|         return await SQLiteHelper.Instance.TableAsync<RoutingItem>().OrderBy(t => t.Sort).ToListAsync(); |         return await SQLiteHelper.Instance.TableAsync<RoutingItem>().OrderBy(t => t.Sort).ToListAsync(); | ||||||
|  |  | ||||||
							
								
								
									
										167
									
								
								v2rayN/ServiceLib/Manager/ProfileGroupItemManager.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										167
									
								
								v2rayN/ServiceLib/Manager/ProfileGroupItemManager.cs
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,167 @@ | ||||||
|  | using System.Collections.Concurrent; | ||||||
|  | 
 | ||||||
|  | namespace ServiceLib.Manager; | ||||||
|  | 
 | ||||||
|  | public class ProfileGroupItemManager | ||||||
|  | { | ||||||
|  |     private static readonly Lazy<ProfileGroupItemManager> _instance = new(() => new()); | ||||||
|  |     private ConcurrentDictionary<string, ProfileGroupItem> _items = new(); | ||||||
|  | 
 | ||||||
|  |     public static ProfileGroupItemManager Instance => _instance.Value; | ||||||
|  |     private static readonly string _tag = "ProfileGroupItemManager"; | ||||||
|  | 
 | ||||||
|  |     private ProfileGroupItemManager() | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public async Task Init() | ||||||
|  |     { | ||||||
|  |         await InitData(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // Read-only getters: do not create or mark dirty | ||||||
|  |     public bool TryGet(string indexId, out ProfileGroupItem? item) | ||||||
|  |     { | ||||||
|  |         item = null; | ||||||
|  |         if (string.IsNullOrWhiteSpace(indexId)) | ||||||
|  |         { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return _items.TryGetValue(indexId, out item); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public ProfileGroupItem? GetOrDefault(string indexId) | ||||||
|  |     { | ||||||
|  |         return string.IsNullOrWhiteSpace(indexId) ? null : (_items.TryGetValue(indexId, out var v) ? v : null); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private async Task InitData() | ||||||
|  |     { | ||||||
|  |         await SQLiteHelper.Instance.ExecuteAsync($"delete from ProfileGroupItem where parentIndexId not in ( select indexId from ProfileItem )"); | ||||||
|  | 
 | ||||||
|  |         var list = await SQLiteHelper.Instance.TableAsync<ProfileGroupItem>().ToListAsync(); | ||||||
|  |         _items = new ConcurrentDictionary<string, ProfileGroupItem>(list.Where(t => !string.IsNullOrEmpty(t.ParentIndexId)).ToDictionary(t => t.ParentIndexId!)); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private ProfileGroupItem AddProfileGroupItem(string indexId) | ||||||
|  |     { | ||||||
|  |         var profileGroupItem = new ProfileGroupItem() | ||||||
|  |         { | ||||||
|  |             ParentIndexId = indexId, | ||||||
|  |             ChildItems = string.Empty, | ||||||
|  |             MultipleLoad = EMultipleLoad.LeastPing | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         _items[indexId] = profileGroupItem; | ||||||
|  |         return profileGroupItem; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private ProfileGroupItem GetProfileGroupItem(string indexId) | ||||||
|  |     { | ||||||
|  |         if (string.IsNullOrEmpty(indexId)) | ||||||
|  |         { | ||||||
|  |             indexId = Utils.GetGuid(false); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return _items.GetOrAdd(indexId, AddProfileGroupItem); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public async Task ClearAll() | ||||||
|  |     { | ||||||
|  |         await SQLiteHelper.Instance.ExecuteAsync($"delete from ProfileGroupItem "); | ||||||
|  |         _items.Clear(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public async Task SaveTo() | ||||||
|  |     { | ||||||
|  |         try | ||||||
|  |         { | ||||||
|  |             var lstExists = await SQLiteHelper.Instance.TableAsync<ProfileGroupItem>().ToListAsync(); | ||||||
|  |             var existsMap = lstExists.Where(t => !string.IsNullOrEmpty(t.ParentIndexId)).ToDictionary(t => t.ParentIndexId!); | ||||||
|  | 
 | ||||||
|  |             var lstInserts = new List<ProfileGroupItem>(); | ||||||
|  |             var lstUpdates = new List<ProfileGroupItem>(); | ||||||
|  | 
 | ||||||
|  |             foreach (var item in _items.Values) | ||||||
|  |             { | ||||||
|  |                 if (string.IsNullOrEmpty(item.ParentIndexId)) | ||||||
|  |                 { | ||||||
|  |                     continue; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 if (existsMap.ContainsKey(item.ParentIndexId)) | ||||||
|  |                 { | ||||||
|  |                     lstUpdates.Add(item); | ||||||
|  |                 } | ||||||
|  |                 else | ||||||
|  |                 { | ||||||
|  |                     lstInserts.Add(item); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             try | ||||||
|  |             { | ||||||
|  |                 if (lstInserts.Count > 0) | ||||||
|  |                 { | ||||||
|  |                     await SQLiteHelper.Instance.InsertAllAsync(lstInserts); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 if (lstUpdates.Count > 0) | ||||||
|  |                 { | ||||||
|  |                     await SQLiteHelper.Instance.UpdateAllAsync(lstUpdates); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             catch (Exception ex) | ||||||
|  |             { | ||||||
|  |                 Logging.SaveLog(_tag, ex); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         catch (Exception ex) | ||||||
|  |         { | ||||||
|  |             Logging.SaveLog(_tag, ex); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public ProfileGroupItem GetOrCreateAndMarkDirty(string indexId) | ||||||
|  |     { | ||||||
|  |         return GetProfileGroupItem(indexId); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public async ValueTask DisposeAsync() | ||||||
|  |     { | ||||||
|  |         await SaveTo(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public async Task SaveItemAsync(ProfileGroupItem item) | ||||||
|  |     { | ||||||
|  |         if (item is null) | ||||||
|  |         { | ||||||
|  |             throw new ArgumentNullException(nameof(item)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (string.IsNullOrWhiteSpace(item.ParentIndexId)) | ||||||
|  |         { | ||||||
|  |             throw new ArgumentException("ParentIndexId required", nameof(item)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         _items[item.ParentIndexId] = item; | ||||||
|  | 
 | ||||||
|  |         try | ||||||
|  |         { | ||||||
|  |             var lst = await SQLiteHelper.Instance.TableAsync<ProfileGroupItem>().Where(t => t.ParentIndexId == item.ParentIndexId).ToListAsync(); | ||||||
|  |             if (lst != null && lst.Count > 0) | ||||||
|  |             { | ||||||
|  |                 await SQLiteHelper.Instance.UpdateAllAsync(new List<ProfileGroupItem> { item }); | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 await SQLiteHelper.Instance.InsertAllAsync(new List<ProfileGroupItem> { item }); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         catch (Exception ex) | ||||||
|  |         { | ||||||
|  |             Logging.SaveLog(_tag, ex); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -35,6 +35,7 @@ public class TaskManager | ||||||
| 
 | 
 | ||||||
|                 await ConfigHandler.SaveConfig(_config); |                 await ConfigHandler.SaveConfig(_config); | ||||||
|                 await ProfileExManager.Instance.SaveTo(); |                 await ProfileExManager.Instance.SaveTo(); | ||||||
|  |                 await ProfileGroupItemManager.Instance.SaveTo(); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             //Execute once 1 hour |             //Execute once 1 hour | ||||||
|  |  | ||||||
							
								
								
									
										13
									
								
								v2rayN/ServiceLib/Models/ProfileGroupItem.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								v2rayN/ServiceLib/Models/ProfileGroupItem.cs
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,13 @@ | ||||||
|  | using SQLite; | ||||||
|  | namespace ServiceLib.Models; | ||||||
|  | 
 | ||||||
|  | [Serializable] | ||||||
|  | public class ProfileGroupItem | ||||||
|  | { | ||||||
|  |     [PrimaryKey] | ||||||
|  |     public string ParentIndexId { get; set; } | ||||||
|  | 
 | ||||||
|  |     public string ChildItems { get; set; } | ||||||
|  | 
 | ||||||
|  |     public EMultipleLoad MultipleLoad { get; set; } = EMultipleLoad.LeastPing; | ||||||
|  | } | ||||||
|  | @ -32,18 +32,21 @@ public class ProfileItem : ReactiveObject | ||||||
|     public string GetSummary() |     public string GetSummary() | ||||||
|     { |     { | ||||||
|         var summary = $"[{(ConfigType).ToString()}] "; |         var summary = $"[{(ConfigType).ToString()}] "; | ||||||
|         var arrAddr = Address.Contains(':') ? Address.Split(':') : Address.Split('.'); |         if (IsComplex()) | ||||||
|         var addr = arrAddr.Length switch |  | ||||||
|         { |         { | ||||||
|             > 2 => $"{arrAddr.First()}***{arrAddr.Last()}", |             summary += $"[{CoreType.ToString()}]{Remarks}"; | ||||||
|             > 1 => $"***{arrAddr.Last()}", |         } | ||||||
|             _ => Address |         else | ||||||
|         }; |  | ||||||
|         summary += ConfigType switch |  | ||||||
|         { |         { | ||||||
|             EConfigType.Custom => $"[{CoreType.ToString()}]{Remarks}", |             var arrAddr = Address.Contains(':') ? Address.Split(':') : Address.Split('.'); | ||||||
|             _ => $"{Remarks}({addr}:{Port})" |             var addr = arrAddr.Length switch | ||||||
|         }; |             { | ||||||
|  |                 > 2 => $"{arrAddr.First()}***{arrAddr.Last()}", | ||||||
|  |                 > 1 => $"***{arrAddr.Last()}", | ||||||
|  |                 _ => Address | ||||||
|  |             }; | ||||||
|  |             summary += $"{Remarks}({addr}:{Port})"; | ||||||
|  |         } | ||||||
|         return summary; |         return summary; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -61,6 +64,51 @@ public class ProfileItem : ReactiveObject | ||||||
|         return Network.TrimEx(); |         return Network.TrimEx(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     public bool IsComplex() | ||||||
|  |     { | ||||||
|  |         return ConfigType is EConfigType.Custom or > EConfigType.Group; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public bool IsValid() | ||||||
|  |     { | ||||||
|  |         if (IsComplex()) | ||||||
|  |             return true; | ||||||
|  | 
 | ||||||
|  |         if (Address.IsNullOrEmpty() || Port is <= 0 or >= 65536) | ||||||
|  |             return false; | ||||||
|  | 
 | ||||||
|  |         switch (ConfigType) | ||||||
|  |         { | ||||||
|  |             case EConfigType.VMess: | ||||||
|  |                 if (Id.IsNullOrEmpty() || !Utils.IsGuidByParse(Id)) | ||||||
|  |                     return false; | ||||||
|  |                 break; | ||||||
|  | 
 | ||||||
|  |             case EConfigType.VLESS: | ||||||
|  |                 if (Id.IsNullOrEmpty() || (!Utils.IsGuidByParse(Id) && Id.Length > 30)) | ||||||
|  |                     return false; | ||||||
|  |                 if (!Global.Flows.Contains(Flow)) | ||||||
|  |                     return false; | ||||||
|  |                 break; | ||||||
|  | 
 | ||||||
|  |             case EConfigType.Shadowsocks: | ||||||
|  |                 if (Id.IsNullOrEmpty()) | ||||||
|  |                     return false; | ||||||
|  |                 if (string.IsNullOrEmpty(Security) || !Global.SsSecuritiesInSingbox.Contains(Security)) | ||||||
|  |                     return false; | ||||||
|  |                 break; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if ((ConfigType is EConfigType.VLESS or EConfigType.Trojan) | ||||||
|  |             && StreamSecurity == Global.StreamSecurityReality | ||||||
|  |             && PublicKey.IsNullOrEmpty()) | ||||||
|  |         { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     #endregion function |     #endregion function | ||||||
| 
 | 
 | ||||||
|     [PrimaryKey] |     [PrimaryKey] | ||||||
|  |  | ||||||
|  | @ -145,6 +145,7 @@ public class Outbound4Sbox : BaseServer4Sbox | ||||||
|     public string? plugin_opts { get; set; } |     public string? plugin_opts { get; set; } | ||||||
|     public List<string>? outbounds { get; set; } |     public List<string>? outbounds { get; set; } | ||||||
|     public bool? interrupt_exist_connections { get; set; } |     public bool? interrupt_exist_connections { get; set; } | ||||||
|  |     public int? tolerance { get; set; } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| public class Endpoints4Sbox : BaseServer4Sbox | public class Endpoints4Sbox : BaseServer4Sbox | ||||||
|  |  | ||||||
							
								
								
									
										252
									
								
								v2rayN/ServiceLib/Resx/ResUI.Designer.cs
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										252
									
								
								v2rayN/ServiceLib/Resx/ResUI.Designer.cs
									
									
									
										generated
									
									
									
								
							|  | @ -672,6 +672,15 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Add Child Configuration 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string menuAddChildServer { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("menuAddChildServer", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Add a custom configuration Configuration 的本地化字符串。 |         ///   查找类似 Add a custom configuration Configuration 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -699,6 +708,24 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Add Policy Group Configuration 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string menuAddPolicyGroupServer { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("menuAddPolicyGroupServer", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Add Proxy Chain Configuration 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string menuAddProxyChainServer { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("menuAddProxyChainServer", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Import Share Links from clipboard (Ctrl+V) 的本地化字符串。 |         ///   查找类似 Import Share Links from clipboard (Ctrl+V) 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -951,6 +978,78 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Generate Policy Group from Multiple Profiles 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string menuGenGroupMultipleServer { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("menuGenGroupMultipleServer", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Multi-Configuration Fallback by sing-box 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string menuGenGroupMultipleServerSingBoxFallback { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("menuGenGroupMultipleServerSingBoxFallback", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Multi-Configuration LeastPing by sing-box 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string menuGenGroupMultipleServerSingBoxLeastPing { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("menuGenGroupMultipleServerSingBoxLeastPing", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Multi-Configuration Fallback by Xray 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string menuGenGroupMultipleServerXrayFallback { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("menuGenGroupMultipleServerXrayFallback", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Multi-Configuration LeastLoad by Xray 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string menuGenGroupMultipleServerXrayLeastLoad { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("menuGenGroupMultipleServerXrayLeastLoad", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Multi-Configuration LeastPing by Xray 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string menuGenGroupMultipleServerXrayLeastPing { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("menuGenGroupMultipleServerXrayLeastPing", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Multi-Configuration Random by Xray 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string menuGenGroupMultipleServerXrayRandom { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("menuGenGroupMultipleServerXrayRandom", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Multi-Configuration RoundRobin by Xray 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string menuGenGroupMultipleServerXrayRoundRobin { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("menuGenGroupMultipleServerXrayRoundRobin", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Global Hotkey Setting 的本地化字符串。 |         ///   查找类似 Global Hotkey Setting 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -1320,6 +1419,15 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Remove Child Configuration 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string menuRemoveChildServer { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("menuRemoveChildServer", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Remove duplicate Configurations 的本地化字符串。 |         ///   查找类似 Remove duplicate Configurations 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -1473,6 +1581,15 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Server List 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string menuServerList { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("menuServerList", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Configurations 的本地化字符串。 |         ///   查找类似 Configurations 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -1482,60 +1599,6 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|         /// <summary> |  | ||||||
|         ///   查找类似 Multi-Configuration to custom configuration 的本地化字符串。 |  | ||||||
|         /// </summary> |  | ||||||
|         public static string menuSetDefaultMultipleServer { |  | ||||||
|             get { |  | ||||||
|                 return ResourceManager.GetString("menuSetDefaultMultipleServer", resourceCulture); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         /// <summary> |  | ||||||
|         ///   查找类似 Multi-Configuration LeastPing by sing-box 的本地化字符串。 |  | ||||||
|         /// </summary> |  | ||||||
|         public static string menuSetDefaultMultipleServerSingBoxLeastPing { |  | ||||||
|             get { |  | ||||||
|                 return ResourceManager.GetString("menuSetDefaultMultipleServerSingBoxLeastPing", resourceCulture); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         /// <summary> |  | ||||||
|         ///   查找类似 Multi-Configuration LeastLoad by Xray 的本地化字符串。 |  | ||||||
|         /// </summary> |  | ||||||
|         public static string menuSetDefaultMultipleServerXrayLeastLoad { |  | ||||||
|             get { |  | ||||||
|                 return ResourceManager.GetString("menuSetDefaultMultipleServerXrayLeastLoad", resourceCulture); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         /// <summary> |  | ||||||
|         ///   查找类似 Multi-Configuration LeastPing by Xray 的本地化字符串。 |  | ||||||
|         /// </summary> |  | ||||||
|         public static string menuSetDefaultMultipleServerXrayLeastPing { |  | ||||||
|             get { |  | ||||||
|                 return ResourceManager.GetString("menuSetDefaultMultipleServerXrayLeastPing", resourceCulture); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         /// <summary> |  | ||||||
|         ///   查找类似 Multi-Configuration Random by Xray 的本地化字符串。 |  | ||||||
|         /// </summary> |  | ||||||
|         public static string menuSetDefaultMultipleServerXrayRandom { |  | ||||||
|             get { |  | ||||||
|                 return ResourceManager.GetString("menuSetDefaultMultipleServerXrayRandom", resourceCulture); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         /// <summary> |  | ||||||
|         ///   查找类似 Multi-Configuration RoundRobin by Xray 的本地化字符串。 |  | ||||||
|         /// </summary> |  | ||||||
|         public static string menuSetDefaultMultipleServerXrayRoundRobin { |  | ||||||
|             get { |  | ||||||
|                 return ResourceManager.GetString("menuSetDefaultMultipleServerXrayRoundRobin", resourceCulture); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Set as active Configuration (Enter) 的本地化字符串。 |         ///   查找类似 Set as active Configuration (Enter) 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -1995,6 +2058,15 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Please Add At Least One Configuration 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string PleaseAddAtLeastOneServer { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("PleaseAddAtLeastOneServer", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Please fill Remarks 的本地化字符串。 |         ///   查找类似 Please fill Remarks 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -2373,6 +2445,24 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Policy Group 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string TbConfigTypePolicyGroup { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("TbConfigTypePolicyGroup", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Proxy Chain 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string TbConfigTypeProxyChain { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("TbConfigTypeProxyChain", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Confirm 的本地化字符串。 |         ///   查找类似 Confirm 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -2526,6 +2616,15 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Fallback 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string TbFallback { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("TbFallback", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Fingerprint 的本地化字符串。 |         ///   查找类似 Fingerprint 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -2643,6 +2742,24 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Most Stable 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string TbLeastLoad { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("TbLeastLoad", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Lowest Latency 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string TbLeastPing { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("TbLeastPing", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Address (IPv4, IPv6) 的本地化字符串。 |         ///   查找类似 Address (IPv4, IPv6) 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -2697,6 +2814,15 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Policy Group Type 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string TbPolicyGroupType { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("TbPolicyGroupType", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 Port 的本地化字符串。 |         ///   查找类似 Port 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -2769,6 +2895,15 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Random 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string TbRandom { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("TbRandom", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 v2ray Full Config Template 的本地化字符串。 |         ///   查找类似 v2ray Full Config Template 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  | @ -2832,6 +2967,15 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Round Robin 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string TbRoundRobin { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("TbRoundRobin", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 socks: local port, socks2: second local port, socks3: LAN port 的本地化字符串。 |         ///   查找类似 socks: local port, socks2: second local port, socks3: LAN port 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  |  | ||||||
|  | @ -1377,22 +1377,22 @@ | ||||||
|   <data name="TbPorts7Tips" xml:space="preserve"> |   <data name="TbPorts7Tips" xml:space="preserve"> | ||||||
|     <value>مخفی و پورت می شود، با کاما (،) جدا می شود</value> |     <value>مخفی و پورت می شود، با کاما (،) جدا می شود</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServer" xml:space="preserve"> |   <data name="menuGenGroupMultipleServer" xml:space="preserve"> | ||||||
|     <value>چند سرور به پیکربندی سفارشی</value> |     <value>Generate Policy Group from Multiple Profiles</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayRandom" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayRandom" xml:space="preserve"> | ||||||
|     <value>چند سرور تصادفی توسط Xray</value> |     <value>چند سرور تصادفی توسط Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayRoundRobin" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayRoundRobin" xml:space="preserve"> | ||||||
|     <value>چند سرور RoundRobin توسط Xray</value> |     <value>چند سرور RoundRobin توسط Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayLeastPing" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayLeastPing" xml:space="preserve"> | ||||||
|     <value>چند سرور LeastPing توسط Xray</value> |     <value>چند سرور LeastPing توسط Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayLeastLoad" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayLeastLoad" xml:space="preserve"> | ||||||
|     <value>چند سرور LeastLoad توسط Xray</value> |     <value>چند سرور LeastLoad توسط Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerSingBoxLeastPing" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerSingBoxLeastPing" xml:space="preserve"> | ||||||
|     <value>LeastPing چند سرور توسط sing-box</value> |     <value>LeastPing چند سرور توسط sing-box</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuExportConfig" xml:space="preserve"> |   <data name="menuExportConfig" xml:space="preserve"> | ||||||
|  | @ -1515,4 +1515,52 @@ | ||||||
|   <data name="TbFakeIPTips" xml:space="preserve"> |   <data name="TbFakeIPTips" xml:space="preserve"> | ||||||
|     <value>Applies globally by default, with built-in FakeIP filtering (sing-box only).</value> |     <value>Applies globally by default, with built-in FakeIP filtering (sing-box only).</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="PleaseAddAtLeastOneServer" xml:space="preserve"> | ||||||
|  |     <value>Please Add At Least One Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbConfigTypePolicyGroup" xml:space="preserve"> | ||||||
|  |     <value>Policy Group</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbConfigTypeProxyChain" xml:space="preserve"> | ||||||
|  |     <value>Proxy Chain</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbLeastPing" xml:space="preserve"> | ||||||
|  |     <value>Lowest Latency</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbRandom" xml:space="preserve"> | ||||||
|  |     <value>Random</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbRoundRobin" xml:space="preserve"> | ||||||
|  |     <value>Round Robin</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbLeastLoad" xml:space="preserve"> | ||||||
|  |     <value>Most Stable</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbPolicyGroupType" xml:space="preserve"> | ||||||
|  |     <value>Policy Group Type</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddPolicyGroupServer" xml:space="preserve"> | ||||||
|  |     <value>Add Policy Group Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddProxyChainServer" xml:space="preserve"> | ||||||
|  |     <value>Add Proxy Chain Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddChildServer" xml:space="preserve"> | ||||||
|  |     <value>Add Child Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuRemoveChildServer" xml:space="preserve"> | ||||||
|  |     <value>Remove Child Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuServerList" xml:space="preserve"> | ||||||
|  |     <value>Server List</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbFallback" xml:space="preserve"> | ||||||
|  |     <value>Fallback</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuGenGroupMultipleServerSingBoxFallback" xml:space="preserve"> | ||||||
|  |     <value>Multi-Configuration Fallback by sing-box</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> | ||||||
|  |     <value>Multi-Configuration Fallback by Xray</value> | ||||||
|  |   </data> | ||||||
| </root> | </root> | ||||||
|  | @ -1377,22 +1377,22 @@ | ||||||
|   <data name="TbPorts7Tips" xml:space="preserve"> |   <data name="TbPorts7Tips" xml:space="preserve"> | ||||||
|     <value>A portot lefedi, vesszővel (,) elválasztva</value> |     <value>A portot lefedi, vesszővel (,) elválasztva</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServer" xml:space="preserve"> |   <data name="menuGenGroupMultipleServer" xml:space="preserve"> | ||||||
|     <value>Több konfiguráció egyéni konfigurációra</value> |     <value>Generate Policy Group from Multiple Profiles</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayRandom" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayRandom" xml:space="preserve"> | ||||||
|     <value>Több konfiguráció véletlenszerűen Xray szerint</value> |     <value>Több konfiguráció véletlenszerűen Xray szerint</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayRoundRobin" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayRoundRobin" xml:space="preserve"> | ||||||
|     <value>Több konfiguráció RoundRobin Xray szerint</value> |     <value>Több konfiguráció RoundRobin Xray szerint</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayLeastPing" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayLeastPing" xml:space="preserve"> | ||||||
|     <value>Több konfiguráció legkisebb pinggel Xray szerint</value> |     <value>Több konfiguráció legkisebb pinggel Xray szerint</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayLeastLoad" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayLeastLoad" xml:space="preserve"> | ||||||
|     <value>Több konfiguráció legkisebb terheléssel Xray szerint</value> |     <value>Több konfiguráció legkisebb terheléssel Xray szerint</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerSingBoxLeastPing" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerSingBoxLeastPing" xml:space="preserve"> | ||||||
|     <value>Több konfiguráció legkisebb pinggel sing-box szerint</value> |     <value>Több konfiguráció legkisebb pinggel sing-box szerint</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuExportConfig" xml:space="preserve"> |   <data name="menuExportConfig" xml:space="preserve"> | ||||||
|  | @ -1515,4 +1515,52 @@ | ||||||
|   <data name="TbFakeIPTips" xml:space="preserve"> |   <data name="TbFakeIPTips" xml:space="preserve"> | ||||||
|     <value>Applies globally by default, with built-in FakeIP filtering (sing-box only).</value> |     <value>Applies globally by default, with built-in FakeIP filtering (sing-box only).</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="PleaseAddAtLeastOneServer" xml:space="preserve"> | ||||||
|  |     <value>Please Add At Least One Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbConfigTypePolicyGroup" xml:space="preserve"> | ||||||
|  |     <value>Policy Group</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbConfigTypeProxyChain" xml:space="preserve"> | ||||||
|  |     <value>Proxy Chain</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbLeastPing" xml:space="preserve"> | ||||||
|  |     <value>Lowest Latency</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbRandom" xml:space="preserve"> | ||||||
|  |     <value>Random</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbRoundRobin" xml:space="preserve"> | ||||||
|  |     <value>Round Robin</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbLeastLoad" xml:space="preserve"> | ||||||
|  |     <value>Most Stable</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbPolicyGroupType" xml:space="preserve"> | ||||||
|  |     <value>Policy Group Type</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddPolicyGroupServer" xml:space="preserve"> | ||||||
|  |     <value>Add Policy Group Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddProxyChainServer" xml:space="preserve"> | ||||||
|  |     <value>Add Proxy Chain Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddChildServer" xml:space="preserve"> | ||||||
|  |     <value>Add Child Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuRemoveChildServer" xml:space="preserve"> | ||||||
|  |     <value>Remove Child Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuServerList" xml:space="preserve"> | ||||||
|  |     <value>Server List</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbFallback" xml:space="preserve"> | ||||||
|  |     <value>Fallback</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuGenGroupMultipleServerSingBoxFallback" xml:space="preserve"> | ||||||
|  |     <value>Multi-Configuration Fallback by sing-box</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> | ||||||
|  |     <value>Multi-Configuration Fallback by Xray</value> | ||||||
|  |   </data> | ||||||
| </root> | </root> | ||||||
|  | @ -1377,22 +1377,22 @@ | ||||||
|   <data name="TbPorts7Tips" xml:space="preserve"> |   <data name="TbPorts7Tips" xml:space="preserve"> | ||||||
|     <value>Will cover the port, separate with commas (,)</value> |     <value>Will cover the port, separate with commas (,)</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServer" xml:space="preserve"> |   <data name="menuGenGroupMultipleServer" xml:space="preserve"> | ||||||
|     <value>Multi-Configuration to custom configuration</value> |     <value>Generate Policy Group from Multiple Profiles</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayRandom" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayRandom" xml:space="preserve"> | ||||||
|     <value>Multi-Configuration Random by Xray</value> |     <value>Multi-Configuration Random by Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayRoundRobin" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayRoundRobin" xml:space="preserve"> | ||||||
|     <value>Multi-Configuration RoundRobin by Xray</value> |     <value>Multi-Configuration RoundRobin by Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayLeastPing" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayLeastPing" xml:space="preserve"> | ||||||
|     <value>Multi-Configuration LeastPing by Xray</value> |     <value>Multi-Configuration LeastPing by Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayLeastLoad" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayLeastLoad" xml:space="preserve"> | ||||||
|     <value>Multi-Configuration LeastLoad by Xray</value> |     <value>Multi-Configuration LeastLoad by Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerSingBoxLeastPing" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerSingBoxLeastPing" xml:space="preserve"> | ||||||
|     <value>Multi-Configuration LeastPing by sing-box</value> |     <value>Multi-Configuration LeastPing by sing-box</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuExportConfig" xml:space="preserve"> |   <data name="menuExportConfig" xml:space="preserve"> | ||||||
|  | @ -1515,4 +1515,52 @@ | ||||||
|   <data name="TbFakeIPTips" xml:space="preserve"> |   <data name="TbFakeIPTips" xml:space="preserve"> | ||||||
|     <value>Applies globally by default, with built-in FakeIP filtering (sing-box only).</value> |     <value>Applies globally by default, with built-in FakeIP filtering (sing-box only).</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="PleaseAddAtLeastOneServer" xml:space="preserve"> | ||||||
|  |     <value>Please Add At Least One Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbConfigTypePolicyGroup" xml:space="preserve"> | ||||||
|  |     <value>Policy Group</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbConfigTypeProxyChain" xml:space="preserve"> | ||||||
|  |     <value>Proxy Chain</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbLeastPing" xml:space="preserve"> | ||||||
|  |     <value>Lowest Latency</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbRandom" xml:space="preserve"> | ||||||
|  |     <value>Random</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbRoundRobin" xml:space="preserve"> | ||||||
|  |     <value>Round Robin</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbLeastLoad" xml:space="preserve"> | ||||||
|  |     <value>Most Stable</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbPolicyGroupType" xml:space="preserve"> | ||||||
|  |     <value>Policy Group Type</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddPolicyGroupServer" xml:space="preserve"> | ||||||
|  |     <value>Add Policy Group Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddProxyChainServer" xml:space="preserve"> | ||||||
|  |     <value>Add Proxy Chain Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddChildServer" xml:space="preserve"> | ||||||
|  |     <value>Add Child Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuRemoveChildServer" xml:space="preserve"> | ||||||
|  |     <value>Remove Child Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuServerList" xml:space="preserve"> | ||||||
|  |     <value>Server List</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbFallback" xml:space="preserve"> | ||||||
|  |     <value>Fallback</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuGenGroupMultipleServerSingBoxFallback" xml:space="preserve"> | ||||||
|  |     <value>Multi-Configuration Fallback by sing-box</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> | ||||||
|  |     <value>Multi-Configuration Fallback by Xray</value> | ||||||
|  |   </data> | ||||||
| </root> | </root> | ||||||
|  | @ -1377,22 +1377,22 @@ | ||||||
|   <data name="TbPorts7Tips" xml:space="preserve"> |   <data name="TbPorts7Tips" xml:space="preserve"> | ||||||
|     <value>Заменит указанный порт, перечисляйте через запятую (,)</value> |     <value>Заменит указанный порт, перечисляйте через запятую (,)</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServer" xml:space="preserve"> |   <data name="menuGenGroupMultipleServer" xml:space="preserve"> | ||||||
|     <value>От мультиконфигурации к пользовательской конфигурации</value> |     <value>Generate Policy Group from Multiple Profiles</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayRandom" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayRandom" xml:space="preserve"> | ||||||
|     <value>Случайный (Xray)</value> |     <value>Случайный (Xray)</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayRoundRobin" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayRoundRobin" xml:space="preserve"> | ||||||
|     <value>Круговой (Xray)</value> |     <value>Круговой (Xray)</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayLeastPing" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayLeastPing" xml:space="preserve"> | ||||||
|     <value>Минимальное RTT (минимальное время туда-обратно) (Xray)</value> |     <value>Минимальное RTT (минимальное время туда-обратно) (Xray)</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayLeastLoad" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayLeastLoad" xml:space="preserve"> | ||||||
|     <value>Минимальная нагрузка (Xray)</value> |     <value>Минимальная нагрузка (Xray)</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerSingBoxLeastPing" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerSingBoxLeastPing" xml:space="preserve"> | ||||||
|     <value>Минимальное RTT (минимальное время туда-обратно) (sing-box)</value> |     <value>Минимальное RTT (минимальное время туда-обратно) (sing-box)</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuExportConfig" xml:space="preserve"> |   <data name="menuExportConfig" xml:space="preserve"> | ||||||
|  | @ -1515,4 +1515,52 @@ | ||||||
|   <data name="TbFakeIPTips" xml:space="preserve"> |   <data name="TbFakeIPTips" xml:space="preserve"> | ||||||
|     <value>Applies globally by default, with built-in FakeIP filtering (sing-box only).</value> |     <value>Applies globally by default, with built-in FakeIP filtering (sing-box only).</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="PleaseAddAtLeastOneServer" xml:space="preserve"> | ||||||
|  |     <value>Please Add At Least One Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbConfigTypePolicyGroup" xml:space="preserve"> | ||||||
|  |     <value>Policy Group</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbConfigTypeProxyChain" xml:space="preserve"> | ||||||
|  |     <value>Proxy Chain</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbLeastPing" xml:space="preserve"> | ||||||
|  |     <value>Lowest Latency</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbRandom" xml:space="preserve"> | ||||||
|  |     <value>Random</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbRoundRobin" xml:space="preserve"> | ||||||
|  |     <value>Round Robin</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbLeastLoad" xml:space="preserve"> | ||||||
|  |     <value>Most Stable</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbPolicyGroupType" xml:space="preserve"> | ||||||
|  |     <value>Policy Group Type</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddPolicyGroupServer" xml:space="preserve"> | ||||||
|  |     <value>Add Policy Group Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddProxyChainServer" xml:space="preserve"> | ||||||
|  |     <value>Add Proxy Chain Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddChildServer" xml:space="preserve"> | ||||||
|  |     <value>Add Child Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuRemoveChildServer" xml:space="preserve"> | ||||||
|  |     <value>Remove Child Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuServerList" xml:space="preserve"> | ||||||
|  |     <value>Server List</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbFallback" xml:space="preserve"> | ||||||
|  |     <value>Fallback</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuGenGroupMultipleServerSingBoxFallback" xml:space="preserve"> | ||||||
|  |     <value>Multi-Configuration Fallback by sing-box</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> | ||||||
|  |     <value>Multi-Configuration Fallback by Xray</value> | ||||||
|  |   </data> | ||||||
| </root> | </root> | ||||||
|  | @ -1374,22 +1374,22 @@ | ||||||
|   <data name="TbPorts7Tips" xml:space="preserve"> |   <data name="TbPorts7Tips" xml:space="preserve"> | ||||||
|     <value>会覆盖端口,多组时用逗号 (,) 隔开</value> |     <value>会覆盖端口,多组时用逗号 (,) 隔开</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServer" xml:space="preserve"> |   <data name="menuGenGroupMultipleServer" xml:space="preserve"> | ||||||
|     <value>多配置文件产生自定义配置 (多选)</value> |     <value>多配置文件生成策略组</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayRandom" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayRandom" xml:space="preserve"> | ||||||
|     <value>多配置文件随机 Xray</value> |     <value>多配置文件随机 Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayRoundRobin" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayRoundRobin" xml:space="preserve"> | ||||||
|     <value>多配置文件负载均衡 Xray</value> |     <value>多配置文件负载均衡 Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayLeastPing" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayLeastPing" xml:space="preserve"> | ||||||
|     <value>多配置文件最低延迟 Xray</value> |     <value>多配置文件最低延迟 Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayLeastLoad" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayLeastLoad" xml:space="preserve"> | ||||||
|     <value>多配置文件最稳定 Xray</value> |     <value>多配置文件最稳定 Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerSingBoxLeastPing" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerSingBoxLeastPing" xml:space="preserve"> | ||||||
|     <value>多配置文件最低延迟 sing-box</value> |     <value>多配置文件最低延迟 sing-box</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuExportConfig" xml:space="preserve"> |   <data name="menuExportConfig" xml:space="preserve"> | ||||||
|  | @ -1512,4 +1512,52 @@ | ||||||
|   <data name="TbFakeIPTips" xml:space="preserve"> |   <data name="TbFakeIPTips" xml:space="preserve"> | ||||||
|     <value>默认全局生效,内置 FakeIP 过滤,仅在 sing-box 中生效</value> |     <value>默认全局生效,内置 FakeIP 过滤,仅在 sing-box 中生效</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="PleaseAddAtLeastOneServer" xml:space="preserve"> | ||||||
|  |     <value>请至少添加一个配置文件</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbConfigTypePolicyGroup" xml:space="preserve"> | ||||||
|  |     <value>策略组</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbConfigTypeProxyChain" xml:space="preserve"> | ||||||
|  |     <value>链式代理</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbLeastPing" xml:space="preserve"> | ||||||
|  |     <value>最低延迟</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbRandom" xml:space="preserve"> | ||||||
|  |     <value>随机</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbRoundRobin" xml:space="preserve"> | ||||||
|  |     <value>负载均衡</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbLeastLoad" xml:space="preserve"> | ||||||
|  |     <value>最稳定</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbPolicyGroupType" xml:space="preserve"> | ||||||
|  |     <value>策略组类型</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddPolicyGroupServer" xml:space="preserve"> | ||||||
|  |     <value>添加策略组配置文件</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddProxyChainServer" xml:space="preserve"> | ||||||
|  |     <value>添加链式代理配置文件</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddChildServer" xml:space="preserve"> | ||||||
|  |     <value>添加子配置文件</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuRemoveChildServer" xml:space="preserve"> | ||||||
|  |     <value>删除子配置文件</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuServerList" xml:space="preserve"> | ||||||
|  |     <value>服务器列表</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbFallback" xml:space="preserve"> | ||||||
|  |     <value>故障转移</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuGenGroupMultipleServerSingBoxFallback" xml:space="preserve"> | ||||||
|  |     <value>多配置文件故障转移 sing-box</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> | ||||||
|  |     <value>多配置文件故障转移 Xray</value> | ||||||
|  |   </data> | ||||||
| </root> | </root> | ||||||
|  | @ -1374,22 +1374,22 @@ | ||||||
|   <data name="TbPorts7Tips" xml:space="preserve"> |   <data name="TbPorts7Tips" xml:space="preserve"> | ||||||
|     <value>會覆蓋埠,多組時用逗號 (,) 隔開</value> |     <value>會覆蓋埠,多組時用逗號 (,) 隔開</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServer" xml:space="preserve"> |   <data name="menuGenGroupMultipleServer" xml:space="preserve"> | ||||||
|     <value>多設定檔產生自訂配置 (多選)</value> |     <value>Generate Policy Group from Multiple Profiles</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayRandom" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayRandom" xml:space="preserve"> | ||||||
|     <value>多設定檔隨機 Xray</value> |     <value>多設定檔隨機 Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayRoundRobin" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayRoundRobin" xml:space="preserve"> | ||||||
|     <value>多設定檔負載平衡 Xray</value> |     <value>多設定檔負載平衡 Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayLeastPing" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayLeastPing" xml:space="preserve"> | ||||||
|     <value>多設定檔最低延遲 Xray</value> |     <value>多設定檔最低延遲 Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerXrayLeastLoad" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerXrayLeastLoad" xml:space="preserve"> | ||||||
|     <value>多設定檔最穩定 Xray</value> |     <value>多設定檔最穩定 Xray</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuSetDefaultMultipleServerSingBoxLeastPing" xml:space="preserve"> |   <data name="menuGenGroupMultipleServerSingBoxLeastPing" xml:space="preserve"> | ||||||
|     <value>多設定檔最低延遲 sing-box</value> |     <value>多設定檔最低延遲 sing-box</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="menuExportConfig" xml:space="preserve"> |   <data name="menuExportConfig" xml:space="preserve"> | ||||||
|  | @ -1512,4 +1512,52 @@ | ||||||
|   <data name="TbFakeIPTips" xml:space="preserve"> |   <data name="TbFakeIPTips" xml:space="preserve"> | ||||||
|     <value>Applies globally by default, with built-in FakeIP filtering (sing-box only).</value> |     <value>Applies globally by default, with built-in FakeIP filtering (sing-box only).</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="PleaseAddAtLeastOneServer" xml:space="preserve"> | ||||||
|  |     <value>Please Add At Least One Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbConfigTypePolicyGroup" xml:space="preserve"> | ||||||
|  |     <value>Policy Group</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbConfigTypeProxyChain" xml:space="preserve"> | ||||||
|  |     <value>Proxy Chain</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbLeastPing" xml:space="preserve"> | ||||||
|  |     <value>Lowest Latency</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbRandom" xml:space="preserve"> | ||||||
|  |     <value>Random</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbRoundRobin" xml:space="preserve"> | ||||||
|  |     <value>Round Robin</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbLeastLoad" xml:space="preserve"> | ||||||
|  |     <value>Most Stable</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbPolicyGroupType" xml:space="preserve"> | ||||||
|  |     <value>Policy Group Type</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddPolicyGroupServer" xml:space="preserve"> | ||||||
|  |     <value>Add Policy Group Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddProxyChainServer" xml:space="preserve"> | ||||||
|  |     <value>Add Proxy Chain Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuAddChildServer" xml:space="preserve"> | ||||||
|  |     <value>Add Child Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuRemoveChildServer" xml:space="preserve"> | ||||||
|  |     <value>Remove Child Configuration</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuServerList" xml:space="preserve"> | ||||||
|  |     <value>Server List</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="TbFallback" xml:space="preserve"> | ||||||
|  |     <value>Fallback</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuGenGroupMultipleServerSingBoxFallback" xml:space="preserve"> | ||||||
|  |     <value>Multi-Configuration Fallback by sing-box</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuGenGroupMultipleServerXrayFallback" xml:space="preserve"> | ||||||
|  |     <value>Multi-Configuration Fallback by Xray</value> | ||||||
|  |   </data> | ||||||
| </root> | </root> | ||||||
|  | @ -16,7 +16,7 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|             if (node == null |             if (node == null | ||||||
|                 || node.Port <= 0) |                 || !node.IsValid()) | ||||||
|             { |             { | ||||||
|                 ret.Msg = ResUI.CheckServerSettings; |                 ret.Msg = ResUI.CheckServerSettings; | ||||||
|                 return ret; |                 return ret; | ||||||
|  | @ -29,6 +29,17 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
| 
 | 
 | ||||||
|             ret.Msg = ResUI.InitialConfiguration; |             ret.Msg = ResUI.InitialConfiguration; | ||||||
|              |              | ||||||
|  |             if (node?.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) | ||||||
|  |             { | ||||||
|  |                 switch (node.ConfigType) | ||||||
|  |                 { | ||||||
|  |                     case EConfigType.PolicyGroup: | ||||||
|  |                         return await GenerateClientMultipleLoadConfig(node); | ||||||
|  |                     case EConfigType.ProxyChain: | ||||||
|  |                         return await GenerateClientChainConfig(node); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|             var result = EmbedUtils.GetEmbedText(Global.SingboxSampleClient); |             var result = EmbedUtils.GetEmbedText(Global.SingboxSampleClient); | ||||||
|             if (result.IsNullOrEmpty()) |             if (result.IsNullOrEmpty()) | ||||||
|             { |             { | ||||||
|  | @ -142,12 +153,9 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
|                     continue; |                     continue; | ||||||
|                 } |                 } | ||||||
|                 var item = await AppManager.Instance.GetProfileItem(it.IndexId); |                 var item = await AppManager.Instance.GetProfileItem(it.IndexId); | ||||||
|                 if (it.ConfigType is EConfigType.VMess or EConfigType.VLESS) |                 if (item is null || item.IsComplex() || !item.IsValid()) | ||||||
|                 { |                 { | ||||||
|                     if (item is null || item.Id.IsNullOrEmpty() || !Utils.IsGuidByParse(item.Id)) |                     continue; | ||||||
|                     { |  | ||||||
|                         continue; |  | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 //find unused port |                 //find unused port | ||||||
|  | @ -187,27 +195,6 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
|                 singboxConfig.inbounds.Add(inbound); |                 singboxConfig.inbounds.Add(inbound); | ||||||
| 
 | 
 | ||||||
|                 //outbound |                 //outbound | ||||||
|                 if (item is null) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.Shadowsocks |  | ||||||
|                     && !Global.SsSecuritiesInSingbox.Contains(item.Security)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.VLESS |  | ||||||
|                  && !Global.Flows.Contains(item.Flow)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.ConfigType is EConfigType.VLESS or EConfigType.Trojan |  | ||||||
|                     && item.StreamSecurity == Global.StreamSecurityReality |  | ||||||
|                     && item.PublicKey.IsNullOrEmpty()) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 var server = await GenServer(item); |                 var server = await GenServer(item); | ||||||
|                 if (server is null) |                 if (server is null) | ||||||
|                 { |                 { | ||||||
|  | @ -266,7 +253,8 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
|         var ret = new RetResult(); |         var ret = new RetResult(); | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|             if (node is not { Port: > 0 }) |             if (node == null | ||||||
|  |                 || !node.IsValid()) | ||||||
|             { |             { | ||||||
|                 ret.Msg = ResUI.CheckServerSettings; |                 ret.Msg = ResUI.CheckServerSettings; | ||||||
|                 return ret; |                 return ret; | ||||||
|  | @ -344,7 +332,64 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public async Task<RetResult> GenerateClientMultipleLoadConfig(List<ProfileItem> selecteds) |     public async Task<RetResult> GenerateClientMultipleLoadConfig(ProfileItem parentNode) | ||||||
|  |     { | ||||||
|  |         var ret = new RetResult(); | ||||||
|  |         try | ||||||
|  |         { | ||||||
|  |             if (_config == null) | ||||||
|  |             { | ||||||
|  |                 ret.Msg = ResUI.CheckServerSettings; | ||||||
|  |                 return ret; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             ret.Msg = ResUI.InitialConfiguration; | ||||||
|  | 
 | ||||||
|  |             var result = EmbedUtils.GetEmbedText(Global.SingboxSampleClient); | ||||||
|  |             var txtOutbound = EmbedUtils.GetEmbedText(Global.SingboxSampleOutbound); | ||||||
|  |             if (result.IsNullOrEmpty() || txtOutbound.IsNullOrEmpty()) | ||||||
|  |             { | ||||||
|  |                 ret.Msg = ResUI.FailedGetDefaultConfiguration; | ||||||
|  |                 return ret; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var singboxConfig = JsonUtils.Deserialize<SingboxConfig>(result); | ||||||
|  |             if (singboxConfig == null) | ||||||
|  |             { | ||||||
|  |                 ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||||
|  |                 return ret; | ||||||
|  |             } | ||||||
|  |             singboxConfig.outbounds.RemoveAt(0); | ||||||
|  | 
 | ||||||
|  |             await GenLog(singboxConfig); | ||||||
|  |             await GenInbounds(singboxConfig); | ||||||
|  |             await GenRouting(singboxConfig); | ||||||
|  |             await GenExperimental(singboxConfig); | ||||||
|  | 
 | ||||||
|  |             var groupRet = await GenGroupOutbound(parentNode, singboxConfig); | ||||||
|  |             if (groupRet != 0) | ||||||
|  |             { | ||||||
|  |                 ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||||
|  |                 return ret; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             await GenDns(null, singboxConfig); | ||||||
|  |             await ConvertGeo2Ruleset(singboxConfig); | ||||||
|  | 
 | ||||||
|  |             ret.Success = true; | ||||||
|  | 
 | ||||||
|  |             ret.Data = await ApplyFullConfigTemplate(singboxConfig); | ||||||
|  |             return ret; | ||||||
|  |         } | ||||||
|  |         catch (Exception ex) | ||||||
|  |         { | ||||||
|  |             Logging.SaveLog(_tag, ex); | ||||||
|  |             ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||||
|  |             return ret; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public async Task<RetResult> GenerateClientChainConfig(ProfileItem parentNode) | ||||||
|     { |     { | ||||||
|         var ret = new RetResult(); |         var ret = new RetResult(); | ||||||
|         try |         try | ||||||
|  | @ -378,48 +423,12 @@ public partial class CoreConfigSingboxService(Config config) | ||||||
|             await GenExperimental(singboxConfig); |             await GenExperimental(singboxConfig); | ||||||
|             singboxConfig.outbounds.RemoveAt(0); |             singboxConfig.outbounds.RemoveAt(0); | ||||||
| 
 | 
 | ||||||
|             var proxyProfiles = new List<ProfileItem>(); |             var groupRet = await GenGroupOutbound(parentNode, singboxConfig); | ||||||
|             foreach (var it in selecteds) |             if (groupRet != 0) | ||||||
|             { |  | ||||||
|                 if (!Global.SingboxSupportConfigType.Contains(it.ConfigType)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.Port <= 0) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 var item = await AppManager.Instance.GetProfileItem(it.IndexId); |  | ||||||
|                 if (item is null) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.ConfigType is EConfigType.VMess or EConfigType.VLESS) |  | ||||||
|                 { |  | ||||||
|                     if (item.Id.IsNullOrEmpty() || !Utils.IsGuidByParse(item.Id)) |  | ||||||
|                     { |  | ||||||
|                         continue; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.Shadowsocks |  | ||||||
|                   && !Global.SsSecuritiesInSingbox.Contains(item.Security)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.VLESS && !Global.Flows.Contains(item.Flow)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 //outbound |  | ||||||
|                 proxyProfiles.Add(item); |  | ||||||
|             } |  | ||||||
|             if (proxyProfiles.Count <= 0) |  | ||||||
|             { |             { | ||||||
|                 ret.Msg = ResUI.FailedGenDefaultConfiguration; |                 ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||||
|                 return ret; |                 return ret; | ||||||
|             } |             } | ||||||
|             await GenOutboundsList(proxyProfiles, singboxConfig); |  | ||||||
| 
 | 
 | ||||||
|             await GenDns(null, singboxConfig); |             await GenDns(null, singboxConfig); | ||||||
|             await ConvertGeo2Ruleset(singboxConfig); |             await ConvertGeo2Ruleset(singboxConfig); | ||||||
|  |  | ||||||
|  | @ -414,16 +414,19 @@ public partial class CoreConfigSingboxService | ||||||
|             return 0; |             return 0; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         var domain = string.Empty; |         List<string> domain = new(); | ||||||
|         if (Utils.IsDomain(node.Address)) // normal outbound |         if (Utils.IsDomain(node.Address)) // normal outbound | ||||||
|         { |         { | ||||||
|             domain = node.Address; |             domain.Add(node.Address); | ||||||
|         } |         } | ||||||
|         else if (node.Address == Global.Loopback && node.SpiderX.IsNotEmpty() && Utils.IsDomain(node.SpiderX)) // Tun2SocksAddress |         if (node.Address == Global.Loopback && node.SpiderX.IsNotEmpty()) // Tun2SocksAddress | ||||||
|         { |         { | ||||||
|             domain = node.SpiderX; |             domain.AddRange(Utils.String2List(node.SpiderX) | ||||||
|  |                 .Where(Utils.IsDomain) | ||||||
|  |                 .Distinct() | ||||||
|  |                 .ToList()); | ||||||
|         } |         } | ||||||
|         if (domain.IsNullOrEmpty()) |         if (domain.Count == 0) | ||||||
|         { |         { | ||||||
|             return 0; |             return 0; | ||||||
|         } |         } | ||||||
|  | @ -432,7 +435,7 @@ public partial class CoreConfigSingboxService | ||||||
|         singboxConfig.dns.rules.Insert(0, new Rule4Sbox |         singboxConfig.dns.rules.Insert(0, new Rule4Sbox | ||||||
|         { |         { | ||||||
|             server = server, |             server = server, | ||||||
|             domain = [domain], |             domain = domain, | ||||||
|         }); |         }); | ||||||
| 
 | 
 | ||||||
|         return await Task.FromResult(0); |         return await Task.FromResult(0); | ||||||
|  |  | ||||||
|  | @ -179,13 +179,21 @@ public partial class CoreConfigSingboxService | ||||||
|             if (node.ConfigType == EConfigType.WireGuard) |             if (node.ConfigType == EConfigType.WireGuard) | ||||||
|             { |             { | ||||||
|                 var endpoint = JsonUtils.Deserialize<Endpoints4Sbox>(txtOutbound); |                 var endpoint = JsonUtils.Deserialize<Endpoints4Sbox>(txtOutbound); | ||||||
|                 await GenEndpoint(node, endpoint); |                 var ret = await GenEndpoint(node, endpoint); | ||||||
|  |                 if (ret != 0) | ||||||
|  |                 { | ||||||
|  |                     return null; | ||||||
|  |                 } | ||||||
|                 return endpoint; |                 return endpoint; | ||||||
|             } |             } | ||||||
|             else |             else | ||||||
|             { |             { | ||||||
|                 var outbound = JsonUtils.Deserialize<Outbound4Sbox>(txtOutbound); |                 var outbound = JsonUtils.Deserialize<Outbound4Sbox>(txtOutbound); | ||||||
|                 await GenOutbound(node, outbound); |                 var ret = await GenOutbound(node, outbound); | ||||||
|  |                 if (ret != 0) | ||||||
|  |                 { | ||||||
|  |                     return null; | ||||||
|  |                 } | ||||||
|                 return outbound; |                 return outbound; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  | @ -196,6 +204,67 @@ public partial class CoreConfigSingboxService | ||||||
|         return await Task.FromResult<BaseServer4Sbox?>(null); |         return await Task.FromResult<BaseServer4Sbox?>(null); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     private async Task<int> GenGroupOutbound(ProfileItem node, SingboxConfig singboxConfig, string baseTagName = Global.ProxyTag, bool ignoreOriginChain = false) | ||||||
|  |     { | ||||||
|  |         try | ||||||
|  |         { | ||||||
|  |             if (node.ConfigType is not (EConfigType.PolicyGroup or EConfigType.ProxyChain)) | ||||||
|  |             { | ||||||
|  |                 return -1; | ||||||
|  |             } | ||||||
|  |             ProfileGroupItemManager.Instance.TryGet(node.IndexId, out var profileGroupItem); | ||||||
|  |             if (profileGroupItem is null || profileGroupItem.ChildItems.IsNullOrEmpty()) | ||||||
|  |             { | ||||||
|  |                 return -1; | ||||||
|  |             } | ||||||
|  |             // remove custom nodes | ||||||
|  |             // remove group nodes for proxy chain | ||||||
|  |             // avoid self-reference | ||||||
|  |             var childProfiles = (await Task.WhenAll( | ||||||
|  |                     Utils.String2List(profileGroupItem.ChildItems) | ||||||
|  |                         .Where(p => !p.IsNullOrEmpty()) | ||||||
|  |                         .Select(AppManager.Instance.GetProfileItem) | ||||||
|  |                 )) | ||||||
|  |                 .Where(p => | ||||||
|  |                     p != null | ||||||
|  |                     && p.IsValid() | ||||||
|  |                     && p.ConfigType != EConfigType.Custom | ||||||
|  |                     && (node.ConfigType == EConfigType.PolicyGroup || p.ConfigType < EConfigType.Group) | ||||||
|  |                     && p.IndexId != node.IndexId | ||||||
|  |                 ) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             if (childProfiles.Count <= 0) | ||||||
|  |             { | ||||||
|  |                 return -1; | ||||||
|  |             } | ||||||
|  |             switch (node.ConfigType) | ||||||
|  |             { | ||||||
|  |                 case EConfigType.PolicyGroup: | ||||||
|  |                     if (ignoreOriginChain) | ||||||
|  |                     { | ||||||
|  |                         await GenOutboundsList(childProfiles, singboxConfig, profileGroupItem.MultipleLoad, baseTagName); | ||||||
|  |                     } | ||||||
|  |                     else | ||||||
|  |                     { | ||||||
|  |                         await GenOutboundsListWithChain(childProfiles, singboxConfig, profileGroupItem.MultipleLoad, baseTagName); | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     break; | ||||||
|  |                 case EConfigType.ProxyChain: | ||||||
|  |                     await GenChainOutboundsList(childProfiles, singboxConfig, baseTagName); | ||||||
|  |                     break; | ||||||
|  |                 default: | ||||||
|  |                     break; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         catch (Exception ex) | ||||||
|  |         { | ||||||
|  |             Logging.SaveLog(_tag, ex); | ||||||
|  |         } | ||||||
|  |         return await Task.FromResult(0); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     private async Task<int> GenOutboundMux(ProfileItem node, Outbound4Sbox outbound) |     private async Task<int> GenOutboundMux(ProfileItem node, Outbound4Sbox outbound) | ||||||
|     { |     { | ||||||
|         try |         try | ||||||
|  | @ -410,7 +479,7 @@ public partial class CoreConfigSingboxService | ||||||
|         return 0; |         return 0; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private async Task<int> GenOutboundsList(List<ProfileItem> nodes, SingboxConfig singboxConfig) |     private async Task<int> GenOutboundsListWithChain(List<ProfileItem> nodes, SingboxConfig singboxConfig, EMultipleLoad multipleLoad, string baseTagName = Global.ProxyTag) | ||||||
|     { |     { | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|  | @ -438,6 +507,38 @@ public partial class CoreConfigSingboxService | ||||||
|             { |             { | ||||||
|                 index++; |                 index++; | ||||||
| 
 | 
 | ||||||
|  |                 if (node.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) | ||||||
|  |                 { | ||||||
|  |                     ProfileGroupItemManager.Instance.TryGet(node.IndexId, out var profileGroupItem); | ||||||
|  |                     if (profileGroupItem == null || profileGroupItem.ChildItems.IsNullOrEmpty()) | ||||||
|  |                     { | ||||||
|  |                         continue; | ||||||
|  |                     } | ||||||
|  |                     var childProfiles = (await Task.WhenAll( | ||||||
|  |                             Utils.String2List(profileGroupItem.ChildItems) | ||||||
|  |                             .Where(p => !p.IsNullOrEmpty()) | ||||||
|  |                             .Select(AppManager.Instance.GetProfileItem) | ||||||
|  |                         )).Where(p => p != null).ToList(); | ||||||
|  |                     if (childProfiles.Count <= 0) | ||||||
|  |                     { | ||||||
|  |                         continue; | ||||||
|  |                     } | ||||||
|  |                     var childBaseTagName = $"{baseTagName}-{index}"; | ||||||
|  |                     var ret = node.ConfigType switch | ||||||
|  |                     { | ||||||
|  |                         EConfigType.PolicyGroup => | ||||||
|  |                             await GenOutboundsListWithChain(childProfiles, singboxConfig, profileGroupItem.MultipleLoad, childBaseTagName), | ||||||
|  |                         EConfigType.ProxyChain => | ||||||
|  |                             await GenChainOutboundsList(childProfiles, singboxConfig, childBaseTagName), | ||||||
|  |                         _ => throw new NotImplementedException() | ||||||
|  |                     }; | ||||||
|  |                     if (ret == 0) | ||||||
|  |                     { | ||||||
|  |                         proxyTags.Add(childBaseTagName); | ||||||
|  |                     } | ||||||
|  |                     continue; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|                 // Handle proxy chain |                 // Handle proxy chain | ||||||
|                 string? prevTag = null; |                 string? prevTag = null; | ||||||
|                 var currentServer = await GenServer(node); |                 var currentServer = await GenServer(node); | ||||||
|  | @ -450,7 +551,7 @@ public partial class CoreConfigSingboxService | ||||||
|                 var subItem = await AppManager.Instance.GetSubItem(node.Subid); |                 var subItem = await AppManager.Instance.GetSubItem(node.Subid); | ||||||
| 
 | 
 | ||||||
|                 // current proxy |                 // current proxy | ||||||
|                 currentServer.tag = $"{Global.ProxyTag}-{index}"; |                 currentServer.tag = $"{baseTagName}-{index}"; | ||||||
|                 proxyTags.Add(currentServer.tag); |                 proxyTags.Add(currentServer.tag); | ||||||
| 
 | 
 | ||||||
|                 if (!node.Subid.IsNullOrEmpty()) |                 if (!node.Subid.IsNullOrEmpty()) | ||||||
|  | @ -467,7 +568,7 @@ public partial class CoreConfigSingboxService | ||||||
|                         { |                         { | ||||||
|                             var prevOutbound = JsonUtils.Deserialize<Outbound4Sbox>(txtOutbound); |                             var prevOutbound = JsonUtils.Deserialize<Outbound4Sbox>(txtOutbound); | ||||||
|                             await GenOutbound(prevNode, prevOutbound); |                             await GenOutbound(prevNode, prevOutbound); | ||||||
|                             prevTag = $"prev-{Global.ProxyTag}-{++prevIndex}"; |                             prevTag = $"prev-{baseTagName}-{++prevIndex}"; | ||||||
|                             prevOutbound.tag = prevTag; |                             prevOutbound.tag = prevTag; | ||||||
|                             prevOutbounds.Add(prevOutbound); |                             prevOutbounds.Add(prevOutbound); | ||||||
|                         } |                         } | ||||||
|  | @ -508,16 +609,21 @@ public partial class CoreConfigSingboxService | ||||||
|                 var outUrltest = new Outbound4Sbox |                 var outUrltest = new Outbound4Sbox | ||||||
|                 { |                 { | ||||||
|                     type = "urltest", |                     type = "urltest", | ||||||
|                     tag = $"{Global.ProxyTag}-auto", |                     tag = $"{baseTagName}-auto", | ||||||
|                     outbounds = proxyTags, |                     outbounds = proxyTags, | ||||||
|                     interrupt_exist_connections = false, |                     interrupt_exist_connections = false, | ||||||
|                 }; |                 }; | ||||||
| 
 | 
 | ||||||
|  |                 if (multipleLoad == EMultipleLoad.Fallback) | ||||||
|  |                 { | ||||||
|  |                     outUrltest.tolerance = 5000; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|                 // Add selector outbound (manual selection) |                 // Add selector outbound (manual selection) | ||||||
|                 var outSelector = new Outbound4Sbox |                 var outSelector = new Outbound4Sbox | ||||||
|                 { |                 { | ||||||
|                     type = "selector", |                     type = "selector", | ||||||
|                     tag = Global.ProxyTag, |                     tag = baseTagName, | ||||||
|                     outbounds = JsonUtils.DeepCopy(proxyTags), |                     outbounds = JsonUtils.DeepCopy(proxyTags), | ||||||
|                     interrupt_exist_connections = false, |                     interrupt_exist_connections = false, | ||||||
|                 }; |                 }; | ||||||
|  | @ -574,4 +680,114 @@ public partial class CoreConfigSingboxService | ||||||
|         } |         } | ||||||
|         return null; |         return null; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     private async Task<int> GenOutboundsList(List<ProfileItem> nodes, SingboxConfig singboxConfig, EMultipleLoad multipleLoad, string baseTagName = Global.ProxyTag) | ||||||
|  |     { | ||||||
|  |         var resultOutbounds = new List<Outbound4Sbox>(); | ||||||
|  |         var resultEndpoints = new List<Endpoints4Sbox>(); // For endpoints | ||||||
|  |         var proxyTags = new List<string>(); // For selector and urltest outbounds | ||||||
|  |         for (var i = 0; i < nodes.Count; i++) | ||||||
|  |         { | ||||||
|  |             var node = nodes[i]; | ||||||
|  |             var server = await GenServer(node); | ||||||
|  |             if (server is null) | ||||||
|  |             { | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |             server.tag = baseTagName + (i + 1).ToString(); | ||||||
|  |             if (server is Endpoints4Sbox endpoint) | ||||||
|  |             { | ||||||
|  |                 resultEndpoints.Add(endpoint); | ||||||
|  |             } | ||||||
|  |             else if (server is Outbound4Sbox outbound) | ||||||
|  |             { | ||||||
|  |                 resultOutbounds.Add(outbound); | ||||||
|  |             } | ||||||
|  |             proxyTags.Add(server.tag); | ||||||
|  |         } | ||||||
|  |         // Add urltest outbound (auto selection based on latency) | ||||||
|  |         if (proxyTags.Count > 0) | ||||||
|  |         { | ||||||
|  |             var outUrltest = new Outbound4Sbox | ||||||
|  |             { | ||||||
|  |                 type = "urltest", | ||||||
|  |                 tag = $"{baseTagName}-auto", | ||||||
|  |                 outbounds = proxyTags, | ||||||
|  |                 interrupt_exist_connections = false, | ||||||
|  |             }; | ||||||
|  |             if (multipleLoad == EMultipleLoad.Fallback) | ||||||
|  |             { | ||||||
|  |                 outUrltest.tolerance = 5000; | ||||||
|  |             } | ||||||
|  |             // Add selector outbound (manual selection) | ||||||
|  |             var outSelector = new Outbound4Sbox | ||||||
|  |             { | ||||||
|  |                 type = "selector", | ||||||
|  |                 tag = baseTagName, | ||||||
|  |                 outbounds = JsonUtils.DeepCopy(proxyTags), | ||||||
|  |                 interrupt_exist_connections = false, | ||||||
|  |             }; | ||||||
|  |             outSelector.outbounds.Insert(0, outUrltest.tag); | ||||||
|  |             // Insert these at the beginning | ||||||
|  |             resultOutbounds.Insert(0, outUrltest); | ||||||
|  |             resultOutbounds.Insert(0, outSelector); | ||||||
|  |         } | ||||||
|  |         singboxConfig.outbounds ??= new(); | ||||||
|  |         resultOutbounds.AddRange(singboxConfig.outbounds); | ||||||
|  |         singboxConfig.outbounds = resultOutbounds; | ||||||
|  |         singboxConfig.endpoints ??= new(); | ||||||
|  |         resultEndpoints.AddRange(singboxConfig.endpoints); | ||||||
|  |         singboxConfig.endpoints = resultEndpoints; | ||||||
|  |         return await Task.FromResult(0); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private async Task<int> GenChainOutboundsList(List<ProfileItem> nodes, SingboxConfig singboxConfig, string baseTagName = Global.ProxyTag) | ||||||
|  |     { | ||||||
|  |         // Based on actual network flow instead of data packets | ||||||
|  |         var nodesReverse = nodes.AsEnumerable().Reverse().ToList(); | ||||||
|  |         var resultOutbounds = new List<Outbound4Sbox>(); | ||||||
|  |         var resultEndpoints = new List<Endpoints4Sbox>(); // For endpoints | ||||||
|  |         for (var i = 0; i < nodesReverse.Count; i++) | ||||||
|  |         { | ||||||
|  |             var node = nodesReverse[i]; | ||||||
|  |             var server = await GenServer(node); | ||||||
|  | 
 | ||||||
|  |             if (server is null) | ||||||
|  |             { | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (i == 0) | ||||||
|  |             { | ||||||
|  |                 server.tag = baseTagName; | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 server.tag = baseTagName + i.ToString(); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (i != nodesReverse.Count - 1) | ||||||
|  |             { | ||||||
|  |                 server.detour = baseTagName + (i + 1).ToString(); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (server is Endpoints4Sbox endpoint) | ||||||
|  |             { | ||||||
|  |                 resultEndpoints.Add(endpoint); | ||||||
|  |             } | ||||||
|  |             else if (server is Outbound4Sbox outbound) | ||||||
|  |             { | ||||||
|  |                 resultOutbounds.Add(outbound); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         singboxConfig.outbounds ??= new(); | ||||||
|  |         resultOutbounds.AddRange(singboxConfig.outbounds); | ||||||
|  |         singboxConfig.outbounds = resultOutbounds; | ||||||
|  | 
 | ||||||
|  |         singboxConfig.endpoints ??= new(); | ||||||
|  |         resultEndpoints.AddRange(singboxConfig.endpoints); | ||||||
|  |         singboxConfig.endpoints = resultEndpoints; | ||||||
|  | 
 | ||||||
|  |         return await Task.FromResult(0); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -368,8 +368,10 @@ public partial class CoreConfigSingboxService | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         var node = await AppManager.Instance.GetProfileItemViaRemarks(outboundTag); |         var node = await AppManager.Instance.GetProfileItemViaRemarks(outboundTag); | ||||||
|  | 
 | ||||||
|         if (node == null |         if (node == null | ||||||
|             || !Global.SingboxSupportConfigType.Contains(node.ConfigType)) |             || (!Global.SingboxSupportConfigType.Contains(node.ConfigType) | ||||||
|  |             && node.ConfigType is not (EConfigType.PolicyGroup or EConfigType.ProxyChain))) | ||||||
|         { |         { | ||||||
|             return Global.ProxyTag; |             return Global.ProxyTag; | ||||||
|         } |         } | ||||||
|  | @ -381,13 +383,24 @@ public partial class CoreConfigSingboxService | ||||||
|             return tag; |             return tag; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         if (node.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) | ||||||
|  |         { | ||||||
|  |             var childBaseTagName = $"{Global.ProxyTag}-{node.IndexId}"; | ||||||
|  |             var ret = await GenGroupOutbound(node, singboxConfig, childBaseTagName); | ||||||
|  |             if (ret == 0) | ||||||
|  |             { | ||||||
|  |                 return childBaseTagName; | ||||||
|  |             } | ||||||
|  |             return Global.ProxyTag; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         var server = await GenServer(node); |         var server = await GenServer(node); | ||||||
|         if (server is null) |         if (server is null) | ||||||
|         { |         { | ||||||
|             return Global.ProxyTag; |             return Global.ProxyTag; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         server.tag = Global.ProxyTag + node.IndexId.ToString(); |         server.tag = tag; | ||||||
|         if (server is Endpoints4Sbox endpoint) |         if (server is Endpoints4Sbox endpoint) | ||||||
|         { |         { | ||||||
|             singboxConfig.endpoints ??= new(); |             singboxConfig.endpoints ??= new(); | ||||||
|  |  | ||||||
|  | @ -16,7 +16,7 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|             if (node == null |             if (node == null | ||||||
|                 || node.Port <= 0) |                 || !node.IsValid()) | ||||||
|             { |             { | ||||||
|                 ret.Msg = ResUI.CheckServerSettings; |                 ret.Msg = ResUI.CheckServerSettings; | ||||||
|                 return ret; |                 return ret; | ||||||
|  | @ -30,6 +30,17 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
| 
 | 
 | ||||||
|             ret.Msg = ResUI.InitialConfiguration; |             ret.Msg = ResUI.InitialConfiguration; | ||||||
| 
 | 
 | ||||||
|  |             if (node?.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) | ||||||
|  |             { | ||||||
|  |                 switch (node.ConfigType) | ||||||
|  |                 { | ||||||
|  |                     case EConfigType.PolicyGroup: | ||||||
|  |                         return await GenerateClientMultipleLoadConfig(node); | ||||||
|  |                     case EConfigType.ProxyChain: | ||||||
|  |                         return await GenerateClientChainConfig(node); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|             var result = EmbedUtils.GetEmbedText(Global.V2raySampleClient); |             var result = EmbedUtils.GetEmbedText(Global.V2raySampleClient); | ||||||
|             if (result.IsNullOrEmpty()) |             if (result.IsNullOrEmpty()) | ||||||
|             { |             { | ||||||
|  | @ -71,7 +82,112 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public async Task<RetResult> GenerateClientMultipleLoadConfig(List<ProfileItem> selecteds, EMultipleLoad multipleLoad) |     public async Task<RetResult> GenerateClientMultipleLoadConfig(ProfileItem parentNode) | ||||||
|  |     { | ||||||
|  |         var ret = new RetResult(); | ||||||
|  | 
 | ||||||
|  |         try | ||||||
|  |         { | ||||||
|  |             if (_config == null) | ||||||
|  |             { | ||||||
|  |                 ret.Msg = ResUI.CheckServerSettings; | ||||||
|  |                 return ret; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             ret.Msg = ResUI.InitialConfiguration; | ||||||
|  | 
 | ||||||
|  |             string result = EmbedUtils.GetEmbedText(Global.V2raySampleClient); | ||||||
|  |             string txtOutbound = EmbedUtils.GetEmbedText(Global.V2raySampleOutbound); | ||||||
|  |             if (result.IsNullOrEmpty() || txtOutbound.IsNullOrEmpty()) | ||||||
|  |             { | ||||||
|  |                 ret.Msg = ResUI.FailedGetDefaultConfiguration; | ||||||
|  |                 return ret; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var v2rayConfig = JsonUtils.Deserialize<V2rayConfig>(result); | ||||||
|  |             if (v2rayConfig == null) | ||||||
|  |             { | ||||||
|  |                 ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||||
|  |                 return ret; | ||||||
|  |             } | ||||||
|  |             v2rayConfig.outbounds.RemoveAt(0); | ||||||
|  | 
 | ||||||
|  |             await GenLog(v2rayConfig); | ||||||
|  |             await GenInbounds(v2rayConfig); | ||||||
|  |             await GenRouting(v2rayConfig); | ||||||
|  |             await GenDns(null, v2rayConfig); | ||||||
|  |             await GenStatistic(v2rayConfig); | ||||||
|  | 
 | ||||||
|  |             var groupRet = await GenGroupOutbound(parentNode, v2rayConfig); | ||||||
|  |             if (groupRet != 0) | ||||||
|  |             { | ||||||
|  |                 ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||||
|  |                 return ret; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var defaultBalancerTag = $"{Global.ProxyTag}{Global.BalancerTagSuffix}"; | ||||||
|  | 
 | ||||||
|  |             //add rule | ||||||
|  |             var rules = v2rayConfig.routing.rules; | ||||||
|  |             if (rules?.Count > 0) | ||||||
|  |             { | ||||||
|  |                 var balancerTagSet = v2rayConfig.routing.balancers | ||||||
|  |                     .Select(b => b.tag) | ||||||
|  |                     .ToHashSet(); | ||||||
|  | 
 | ||||||
|  |                 foreach (var rule in rules) | ||||||
|  |                 { | ||||||
|  |                     if (rule.outboundTag == null) | ||||||
|  |                         continue; | ||||||
|  | 
 | ||||||
|  |                     if (balancerTagSet.Contains(rule.outboundTag)) | ||||||
|  |                     { | ||||||
|  |                         rule.balancerTag = rule.outboundTag; | ||||||
|  |                         rule.outboundTag = null; | ||||||
|  |                         continue; | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     var outboundWithSuffix = rule.outboundTag + Global.BalancerTagSuffix; | ||||||
|  |                     if (balancerTagSet.Contains(outboundWithSuffix)) | ||||||
|  |                     { | ||||||
|  |                         rule.balancerTag = outboundWithSuffix; | ||||||
|  |                         rule.outboundTag = null; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             if (v2rayConfig.routing.domainStrategy == Global.IPIfNonMatch) | ||||||
|  |             { | ||||||
|  |                 v2rayConfig.routing.rules.Add(new() | ||||||
|  |                 { | ||||||
|  |                     ip = ["0.0.0.0/0", "::/0"], | ||||||
|  |                     balancerTag = defaultBalancerTag, | ||||||
|  |                     type = "field" | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 v2rayConfig.routing.rules.Add(new() | ||||||
|  |                 { | ||||||
|  |                     network = "tcp,udp", | ||||||
|  |                     balancerTag = defaultBalancerTag, | ||||||
|  |                     type = "field" | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             ret.Success = true; | ||||||
|  | 
 | ||||||
|  |             ret.Data = await ApplyFullConfigTemplate(v2rayConfig, true); | ||||||
|  |             return ret; | ||||||
|  |         } | ||||||
|  |         catch (Exception ex) | ||||||
|  |         { | ||||||
|  |             Logging.SaveLog(_tag, ex); | ||||||
|  |             ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||||
|  |             return ret; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public async Task<RetResult> GenerateClientChainConfig(ProfileItem parentNode) | ||||||
|     { |     { | ||||||
|         var ret = new RetResult(); |         var ret = new RetResult(); | ||||||
| 
 | 
 | ||||||
|  | @ -107,82 +223,12 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|             await GenStatistic(v2rayConfig); |             await GenStatistic(v2rayConfig); | ||||||
|             v2rayConfig.outbounds.RemoveAt(0); |             v2rayConfig.outbounds.RemoveAt(0); | ||||||
| 
 | 
 | ||||||
|             var proxyProfiles = new List<ProfileItem>(); |             var groupRet = await GenGroupOutbound(parentNode, v2rayConfig); | ||||||
|             foreach (var it in selecteds) |             if (groupRet != 0) | ||||||
|             { |  | ||||||
|                 if (!Global.XraySupportConfigType.Contains(it.ConfigType)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.Port <= 0) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 var item = await AppManager.Instance.GetProfileItem(it.IndexId); |  | ||||||
|                 if (item is null) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.ConfigType is EConfigType.VMess or EConfigType.VLESS) |  | ||||||
|                 { |  | ||||||
|                     if (item.Id.IsNullOrEmpty() || !Utils.IsGuidByParse(item.Id)) |  | ||||||
|                     { |  | ||||||
|                         continue; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.Shadowsocks |  | ||||||
|                   && !Global.SsSecuritiesInSingbox.Contains(item.Security)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.VLESS && !Global.Flows.Contains(item.Flow)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 //outbound |  | ||||||
|                 proxyProfiles.Add(item); |  | ||||||
|             } |  | ||||||
|             if (proxyProfiles.Count <= 0) |  | ||||||
|             { |             { | ||||||
|                 ret.Msg = ResUI.FailedGenDefaultConfiguration; |                 ret.Msg = ResUI.FailedGenDefaultConfiguration; | ||||||
|                 return ret; |                 return ret; | ||||||
|             } |             } | ||||||
|             await GenOutboundsList(proxyProfiles, v2rayConfig); |  | ||||||
| 
 |  | ||||||
|             //add balancers |  | ||||||
|             await GenBalancer(v2rayConfig, multipleLoad); |  | ||||||
| 
 |  | ||||||
|             var balancer = v2rayConfig.routing.balancers.First(); |  | ||||||
| 
 |  | ||||||
|             //add rule |  | ||||||
|             var rules = v2rayConfig.routing.rules.Where(t => t.outboundTag == Global.ProxyTag).ToList(); |  | ||||||
|             if (rules?.Count > 0) |  | ||||||
|             { |  | ||||||
|                 foreach (var rule in rules) |  | ||||||
|                 { |  | ||||||
|                     rule.outboundTag = null; |  | ||||||
|                     rule.balancerTag = balancer.tag; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             if (v2rayConfig.routing.domainStrategy == Global.IPIfNonMatch) |  | ||||||
|             { |  | ||||||
|                 v2rayConfig.routing.rules.Add(new() |  | ||||||
|                 { |  | ||||||
|                     ip = ["0.0.0.0/0", "::/0"], |  | ||||||
|                     balancerTag = balancer.tag, |  | ||||||
|                     type = "field" |  | ||||||
|                 }); |  | ||||||
|             } |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 v2rayConfig.routing.rules.Add(new() |  | ||||||
|                 { |  | ||||||
|                     network = "tcp,udp", |  | ||||||
|                     balancerTag = balancer.tag, |  | ||||||
|                     type = "field" |  | ||||||
|                 }); |  | ||||||
|             } |  | ||||||
| 
 | 
 | ||||||
|             ret.Success = true; |             ret.Success = true; | ||||||
| 
 | 
 | ||||||
|  | @ -255,12 +301,9 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|                     continue; |                     continue; | ||||||
|                 } |                 } | ||||||
|                 var item = await AppManager.Instance.GetProfileItem(it.IndexId); |                 var item = await AppManager.Instance.GetProfileItem(it.IndexId); | ||||||
|                 if (it.ConfigType is EConfigType.VMess or EConfigType.VLESS) |                 if (item is null || item.IsComplex() || !item.IsValid()) | ||||||
|                 { |                 { | ||||||
|                     if (item is null || item.Id.IsNullOrEmpty() || !Utils.IsGuidByParse(item.Id)) |                     continue; | ||||||
|                     { |  | ||||||
|                         continue; |  | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 //find unused port |                 //find unused port | ||||||
|  | @ -289,28 +332,6 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|                 it.Port = port; |                 it.Port = port; | ||||||
|                 it.AllowTest = true; |                 it.AllowTest = true; | ||||||
| 
 | 
 | ||||||
|                 //outbound |  | ||||||
|                 if (item is null) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.Shadowsocks |  | ||||||
|                     && !Global.SsSecuritiesInXray.Contains(item.Security)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (item.ConfigType == EConfigType.VLESS |  | ||||||
|                  && !Global.Flows.Contains(item.Flow)) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
|                 if (it.ConfigType is EConfigType.VLESS or EConfigType.Trojan |  | ||||||
|                     && item.StreamSecurity == Global.StreamSecurityReality |  | ||||||
|                     && item.PublicKey.IsNullOrEmpty()) |  | ||||||
|                 { |  | ||||||
|                     continue; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 //inbound |                 //inbound | ||||||
|                 Inbounds4Ray inbound = new() |                 Inbounds4Ray inbound = new() | ||||||
|                 { |                 { | ||||||
|  | @ -321,6 +342,7 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|                 inbound.tag = inbound.protocol + inbound.port.ToString(); |                 inbound.tag = inbound.protocol + inbound.port.ToString(); | ||||||
|                 v2rayConfig.inbounds.Add(inbound); |                 v2rayConfig.inbounds.Add(inbound); | ||||||
| 
 | 
 | ||||||
|  |                 //outbound | ||||||
|                 var outbound = JsonUtils.Deserialize<Outbounds4Ray>(txtOutbound); |                 var outbound = JsonUtils.Deserialize<Outbounds4Ray>(txtOutbound); | ||||||
|                 await GenOutbound(item, outbound); |                 await GenOutbound(item, outbound); | ||||||
|                 outbound.tag = Global.ProxyTag + inbound.port.ToString(); |                 outbound.tag = Global.ProxyTag + inbound.port.ToString(); | ||||||
|  | @ -354,7 +376,8 @@ public partial class CoreConfigV2rayService(Config config) | ||||||
|         var ret = new RetResult(); |         var ret = new RetResult(); | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|             if (node is not { Port: > 0 }) |             if (node == null | ||||||
|  |                 || !node.IsValid()) | ||||||
|             { |             { | ||||||
|                 ret.Msg = ResUI.CheckServerSettings; |                 ret.Msg = ResUI.CheckServerSettings; | ||||||
|                 return ret; |                 return ret; | ||||||
|  |  | ||||||
|  | @ -2,24 +2,24 @@ namespace ServiceLib.Services.CoreConfig; | ||||||
| 
 | 
 | ||||||
| public partial class CoreConfigV2rayService | public partial class CoreConfigV2rayService | ||||||
| { | { | ||||||
|     private async Task<int> GenBalancer(V2rayConfig v2rayConfig, EMultipleLoad multipleLoad) |     private async Task<int> GenObservatory(V2rayConfig v2rayConfig, EMultipleLoad multipleLoad, string baseTagName = Global.ProxyTag) | ||||||
|     { |     { | ||||||
|         if (multipleLoad == EMultipleLoad.LeastPing) |         if (multipleLoad == EMultipleLoad.LeastPing) | ||||||
|         { |         { | ||||||
|             var observatory = new Observatory4Ray |             var observatory = new Observatory4Ray | ||||||
|             { |             { | ||||||
|                 subjectSelector = [Global.ProxyTag], |                 subjectSelector = [baseTagName], | ||||||
|                 probeUrl = AppManager.Instance.Config.SpeedTestItem.SpeedPingTestUrl, |                 probeUrl = AppManager.Instance.Config.SpeedTestItem.SpeedPingTestUrl, | ||||||
|                 probeInterval = "3m", |                 probeInterval = "3m", | ||||||
|                 enableConcurrency = true, |                 enableConcurrency = true, | ||||||
|             }; |             }; | ||||||
|             v2rayConfig.observatory = observatory; |             v2rayConfig.observatory = observatory; | ||||||
|         } |         } | ||||||
|         else if (multipleLoad == EMultipleLoad.LeastLoad) |         else if (multipleLoad is EMultipleLoad.LeastLoad or EMultipleLoad.Fallback) | ||||||
|         { |         { | ||||||
|             var burstObservatory = new BurstObservatory4Ray |             var burstObservatory = new BurstObservatory4Ray | ||||||
|             { |             { | ||||||
|                 subjectSelector = [Global.ProxyTag], |                 subjectSelector = [baseTagName], | ||||||
|                 pingConfig = new() |                 pingConfig = new() | ||||||
|                 { |                 { | ||||||
|                     destination = AppManager.Instance.Config.SpeedTestItem.SpeedPingTestUrl, |                     destination = AppManager.Instance.Config.SpeedTestItem.SpeedPingTestUrl, | ||||||
|  | @ -30,6 +30,11 @@ public partial class CoreConfigV2rayService | ||||||
|             }; |             }; | ||||||
|             v2rayConfig.burstObservatory = burstObservatory; |             v2rayConfig.burstObservatory = burstObservatory; | ||||||
|         } |         } | ||||||
|  |         return await Task.FromResult(0); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private async Task<string> GenBalancer(V2rayConfig v2rayConfig, EMultipleLoad multipleLoad, string selector = Global.ProxyTag) | ||||||
|  |     { | ||||||
|         var strategyType = multipleLoad switch |         var strategyType = multipleLoad switch | ||||||
|         { |         { | ||||||
|             EMultipleLoad.Random => "random", |             EMultipleLoad.Random => "random", | ||||||
|  | @ -38,13 +43,22 @@ public partial class CoreConfigV2rayService | ||||||
|             EMultipleLoad.LeastLoad => "leastLoad", |             EMultipleLoad.LeastLoad => "leastLoad", | ||||||
|             _ => "roundRobin", |             _ => "roundRobin", | ||||||
|         }; |         }; | ||||||
|  |         var balancerTag = $"{selector}{Global.BalancerTagSuffix}"; | ||||||
|         var balancer = new BalancersItem4Ray |         var balancer = new BalancersItem4Ray | ||||||
|         { |         { | ||||||
|             selector = [Global.ProxyTag], |             selector = [selector], | ||||||
|             strategy = new() { type = strategyType }, |             strategy = new() | ||||||
|             tag = $"{Global.ProxyTag}-round", |             { | ||||||
|  |                 type = strategyType, | ||||||
|  |                 settings = new() | ||||||
|  |                 { | ||||||
|  |                     expected = 1, | ||||||
|  |                 }, | ||||||
|  |             }, | ||||||
|  |             tag = balancerTag, | ||||||
|         }; |         }; | ||||||
|         v2rayConfig.routing.balancers = [balancer]; |         v2rayConfig.routing.balancers ??= new(); | ||||||
|         return await Task.FromResult(0); |         v2rayConfig.routing.balancers.Add(balancer); | ||||||
|  |         return await Task.FromResult(balancerTag); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -480,6 +480,69 @@ public partial class CoreConfigV2rayService | ||||||
|         return 0; |         return 0; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     private async Task<int> GenGroupOutbound(ProfileItem node, V2rayConfig v2rayConfig, string baseTagName = Global.ProxyTag, bool ignoreOriginChain = false) | ||||||
|  |     { | ||||||
|  |         try | ||||||
|  |         { | ||||||
|  |             if (node.ConfigType is not (EConfigType.PolicyGroup or EConfigType.ProxyChain)) | ||||||
|  |             { | ||||||
|  |                 return -1; | ||||||
|  |             } | ||||||
|  |             ProfileGroupItemManager.Instance.TryGet(node.IndexId, out var profileGroupItem); | ||||||
|  |             if (profileGroupItem is null || profileGroupItem.ChildItems.IsNullOrEmpty()) | ||||||
|  |             { | ||||||
|  |                 return -1; | ||||||
|  |             } | ||||||
|  |             // remove custom nodes | ||||||
|  |             // remove group nodes for proxy chain | ||||||
|  |             var childProfiles = (await Task.WhenAll( | ||||||
|  |                     Utils.String2List(profileGroupItem.ChildItems) | ||||||
|  |                         .Where(p => !p.IsNullOrEmpty()) | ||||||
|  |                         .Select(AppManager.Instance.GetProfileItem) | ||||||
|  |                 )) | ||||||
|  |                 .Where(p => | ||||||
|  |                     p != null && | ||||||
|  |                     p.IsValid() && | ||||||
|  |                     p.ConfigType != EConfigType.Custom && | ||||||
|  |                     (node.ConfigType == EConfigType.PolicyGroup || p.ConfigType < EConfigType.Group) | ||||||
|  |                 ) | ||||||
|  |                 .ToList(); | ||||||
|  | 
 | ||||||
|  |             if (childProfiles.Count <= 0) | ||||||
|  |             { | ||||||
|  |                 return -1; | ||||||
|  |             } | ||||||
|  |             switch (node.ConfigType) | ||||||
|  |             { | ||||||
|  |                 case EConfigType.PolicyGroup: | ||||||
|  |                     if (ignoreOriginChain) | ||||||
|  |                     { | ||||||
|  |                         await GenOutboundsList(childProfiles, v2rayConfig, baseTagName); | ||||||
|  |                     } | ||||||
|  |                     else | ||||||
|  |                     { | ||||||
|  |                         await GenOutboundsListWithChain(childProfiles, v2rayConfig, baseTagName); | ||||||
|  |                     } | ||||||
|  |                     break; | ||||||
|  |                 case EConfigType.ProxyChain: | ||||||
|  |                     await GenChainOutboundsList(childProfiles, v2rayConfig, baseTagName); | ||||||
|  |                     break; | ||||||
|  |                 default: | ||||||
|  |                     break; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             //add balancers | ||||||
|  |             await GenObservatory(v2rayConfig, profileGroupItem.MultipleLoad, baseTagName); | ||||||
|  |             await GenBalancer(v2rayConfig, profileGroupItem.MultipleLoad, baseTagName); | ||||||
|  | 
 | ||||||
|  |         } | ||||||
|  |         catch (Exception ex) | ||||||
|  |         { | ||||||
|  |             Logging.SaveLog(_tag, ex); | ||||||
|  |         } | ||||||
|  |         return await Task.FromResult(0); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     private async Task<int> GenMoreOutbounds(ProfileItem node, V2rayConfig v2rayConfig) |     private async Task<int> GenMoreOutbounds(ProfileItem node, V2rayConfig v2rayConfig) | ||||||
|     { |     { | ||||||
|         //fragment proxy |         //fragment proxy | ||||||
|  | @ -552,7 +615,7 @@ public partial class CoreConfigV2rayService | ||||||
|         return 0; |         return 0; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private async Task<int> GenOutboundsList(List<ProfileItem> nodes, V2rayConfig v2rayConfig) |     private async Task<int> GenOutboundsListWithChain(List<ProfileItem> nodes, V2rayConfig v2rayConfig, string baseTagName = Global.ProxyTag) | ||||||
|     { |     { | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|  | @ -577,6 +640,34 @@ public partial class CoreConfigV2rayService | ||||||
|             { |             { | ||||||
|                 index++; |                 index++; | ||||||
| 
 | 
 | ||||||
|  |                 if (node.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) | ||||||
|  |                 { | ||||||
|  |                     ProfileGroupItemManager.Instance.TryGet(node.IndexId, out var profileGroupItem); | ||||||
|  |                     if (profileGroupItem == null || profileGroupItem.ChildItems.IsNullOrEmpty()) | ||||||
|  |                     { | ||||||
|  |                         continue; | ||||||
|  |                     } | ||||||
|  |                     var childProfiles = (await Task.WhenAll( | ||||||
|  |                             Utils.String2List(profileGroupItem.ChildItems) | ||||||
|  |                             .Where(p => !p.IsNullOrEmpty()) | ||||||
|  |                             .Select(AppManager.Instance.GetProfileItem) | ||||||
|  |                         )).Where(p => p != null).ToList(); | ||||||
|  |                     if (childProfiles.Count <= 0) | ||||||
|  |                     { | ||||||
|  |                         continue; | ||||||
|  |                     } | ||||||
|  |                     var childBaseTagName = $"{baseTagName}-{index}"; | ||||||
|  |                     var ret = node.ConfigType switch | ||||||
|  |                     { | ||||||
|  |                         EConfigType.PolicyGroup => | ||||||
|  |                             await GenOutboundsListWithChain(childProfiles, v2rayConfig, childBaseTagName), | ||||||
|  |                         EConfigType.ProxyChain => | ||||||
|  |                             await GenChainOutboundsList(childProfiles, v2rayConfig, childBaseTagName), | ||||||
|  |                         _ => throw new NotImplementedException() | ||||||
|  |                     }; | ||||||
|  |                     continue; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|                 // Handle proxy chain |                 // Handle proxy chain | ||||||
|                 string? prevTag = null; |                 string? prevTag = null; | ||||||
|                 var currentOutbound = JsonUtils.Deserialize<Outbounds4Ray>(txtOutbound); |                 var currentOutbound = JsonUtils.Deserialize<Outbounds4Ray>(txtOutbound); | ||||||
|  | @ -590,7 +681,7 @@ public partial class CoreConfigV2rayService | ||||||
| 
 | 
 | ||||||
|                 // current proxy |                 // current proxy | ||||||
|                 await GenOutbound(node, currentOutbound); |                 await GenOutbound(node, currentOutbound); | ||||||
|                 currentOutbound.tag = $"{Global.ProxyTag}-{index}"; |                 currentOutbound.tag = $"{baseTagName}-{index}"; | ||||||
| 
 | 
 | ||||||
|                 if (!node.Subid.IsNullOrEmpty()) |                 if (!node.Subid.IsNullOrEmpty()) | ||||||
|                 { |                 { | ||||||
|  | @ -606,7 +697,7 @@ public partial class CoreConfigV2rayService | ||||||
|                         { |                         { | ||||||
|                             var prevOutbound = JsonUtils.Deserialize<Outbounds4Ray>(txtOutbound); |                             var prevOutbound = JsonUtils.Deserialize<Outbounds4Ray>(txtOutbound); | ||||||
|                             await GenOutbound(prevNode, prevOutbound); |                             await GenOutbound(prevNode, prevOutbound); | ||||||
|                             prevTag = $"prev-{Global.ProxyTag}-{++prevIndex}"; |                             prevTag = $"prev-{baseTagName}-{++prevIndex}"; | ||||||
|                             prevOutbound.tag = prevTag; |                             prevOutbound.tag = prevTag; | ||||||
|                             prevOutbounds.Add(prevOutbound); |                             prevOutbounds.Add(prevOutbound); | ||||||
|                         } |                         } | ||||||
|  | @ -692,4 +783,78 @@ public partial class CoreConfigV2rayService | ||||||
|         } |         } | ||||||
|         return null; |         return null; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     private async Task<int> GenOutboundsList(List<ProfileItem> nodes, V2rayConfig v2rayConfig, string baseTagName = Global.ProxyTag) | ||||||
|  |     { | ||||||
|  |         var resultOutbounds = new List<Outbounds4Ray>(); | ||||||
|  |         for (var i = 0; i < nodes.Count; i++) | ||||||
|  |         { | ||||||
|  |             var node = nodes[i]; | ||||||
|  |             var txtOutbound = EmbedUtils.GetEmbedText(Global.V2raySampleOutbound); | ||||||
|  |             if (txtOutbound.IsNullOrEmpty()) | ||||||
|  |             { | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |             var outbound = JsonUtils.Deserialize<Outbounds4Ray>(txtOutbound); | ||||||
|  |             var result = await GenOutbound(node, outbound); | ||||||
|  |             if (result != 0) | ||||||
|  |             { | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |             outbound.tag = baseTagName + (i + 1).ToString(); | ||||||
|  |             resultOutbounds.Add(outbound); | ||||||
|  |         } | ||||||
|  |         v2rayConfig.outbounds ??= new(); | ||||||
|  |         resultOutbounds.AddRange(v2rayConfig.outbounds); | ||||||
|  |         v2rayConfig.outbounds = resultOutbounds; | ||||||
|  |         return await Task.FromResult(0); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private async Task<int> GenChainOutboundsList(List<ProfileItem> nodes, V2rayConfig v2RayConfig, string baseTagName = Global.ProxyTag) | ||||||
|  |     { | ||||||
|  |         // Based on actual network flow instead of data packets | ||||||
|  |         var nodesReverse = nodes.AsEnumerable().Reverse().ToList(); | ||||||
|  |         var resultOutbounds = new List<Outbounds4Ray>(); | ||||||
|  |         for (var i = 0; i < nodesReverse.Count; i++) | ||||||
|  |         { | ||||||
|  |             var node = nodesReverse[i]; | ||||||
|  |             var txtOutbound = EmbedUtils.GetEmbedText(Global.V2raySampleOutbound); | ||||||
|  |             if (txtOutbound.IsNullOrEmpty()) | ||||||
|  |             { | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |             var outbound = JsonUtils.Deserialize<Outbounds4Ray>(txtOutbound); | ||||||
|  |             var result = await GenOutbound(node, outbound); | ||||||
|  | 
 | ||||||
|  |             if (result != 0) | ||||||
|  |             { | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (i == 0) | ||||||
|  |             { | ||||||
|  |                 outbound.tag = baseTagName; | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 // avoid v2ray observe | ||||||
|  |                 outbound.tag = "chain-" + baseTagName + i.ToString(); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (i != nodesReverse.Count - 1) | ||||||
|  |             { | ||||||
|  |                 outbound.streamSettings.sockopt = new() | ||||||
|  |                 { | ||||||
|  |                     dialerProxy = "chain-" + baseTagName + (i + 1).ToString() | ||||||
|  |                 }; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             resultOutbounds.Add(outbound); | ||||||
|  |         } | ||||||
|  |         v2RayConfig.outbounds ??= new(); | ||||||
|  |         resultOutbounds.AddRange(v2RayConfig.outbounds); | ||||||
|  |         v2RayConfig.outbounds = resultOutbounds; | ||||||
|  | 
 | ||||||
|  |         return await Task.FromResult(0); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -125,8 +125,10 @@ public partial class CoreConfigV2rayService | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         var node = await AppManager.Instance.GetProfileItemViaRemarks(outboundTag); |         var node = await AppManager.Instance.GetProfileItemViaRemarks(outboundTag); | ||||||
|  | 
 | ||||||
|         if (node == null |         if (node == null | ||||||
|             || !Global.XraySupportConfigType.Contains(node.ConfigType)) |             || (!Global.XraySupportConfigType.Contains(node.ConfigType) | ||||||
|  |             && node.ConfigType is not (EConfigType.PolicyGroup or EConfigType.ProxyChain))) | ||||||
|         { |         { | ||||||
|             return Global.ProxyTag; |             return Global.ProxyTag; | ||||||
|         } |         } | ||||||
|  | @ -137,6 +139,17 @@ public partial class CoreConfigV2rayService | ||||||
|             return tag; |             return tag; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         if (node.ConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) | ||||||
|  |         { | ||||||
|  |             var childBaseTagName = $"{Global.ProxyTag}-{node.IndexId}"; | ||||||
|  |             var ret = await GenGroupOutbound(node, v2rayConfig, childBaseTagName); | ||||||
|  |             if (ret == 0) | ||||||
|  |             { | ||||||
|  |                 return childBaseTagName; | ||||||
|  |             } | ||||||
|  |             return Global.ProxyTag; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         var txtOutbound = EmbedUtils.GetEmbedText(Global.V2raySampleOutbound); |         var txtOutbound = EmbedUtils.GetEmbedText(Global.V2raySampleOutbound); | ||||||
|         var outbound = JsonUtils.Deserialize<Outbounds4Ray>(txtOutbound); |         var outbound = JsonUtils.Deserialize<Outbounds4Ray>(txtOutbound); | ||||||
|         await GenOutbound(node, outbound); |         await GenOutbound(node, outbound); | ||||||
|  |  | ||||||
							
								
								
									
										225
									
								
								v2rayN/ServiceLib/ViewModels/AddGroupServerViewModel.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										225
									
								
								v2rayN/ServiceLib/ViewModels/AddGroupServerViewModel.cs
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,225 @@ | ||||||
|  | using System.Reactive; | ||||||
|  | using DynamicData.Binding; | ||||||
|  | using ReactiveUI; | ||||||
|  | using ReactiveUI.Fody.Helpers; | ||||||
|  | 
 | ||||||
|  | namespace ServiceLib.ViewModels; | ||||||
|  | 
 | ||||||
|  | public class AddGroupServerViewModel : MyReactiveObject | ||||||
|  | { | ||||||
|  |     [Reactive] | ||||||
|  |     public ProfileItem SelectedSource { get; set; } | ||||||
|  | 
 | ||||||
|  |     [Reactive] | ||||||
|  |     public ProfileItem SelectedChild { get; set; } | ||||||
|  | 
 | ||||||
|  |     [Reactive] | ||||||
|  |     public IList<ProfileItem> SelectedChildren { get; set; } | ||||||
|  | 
 | ||||||
|  |     [Reactive] | ||||||
|  |     public string? CoreType { get; set; } | ||||||
|  | 
 | ||||||
|  |     [Reactive] | ||||||
|  |     public string? PolicyGroupType { get; set; } | ||||||
|  | 
 | ||||||
|  |     public IObservableCollection<ProfileItem> ChildItemsObs { get; } = new ObservableCollectionExtended<ProfileItem>(); | ||||||
|  | 
 | ||||||
|  |     //public ReactiveCommand<Unit, Unit> AddCmd { get; } | ||||||
|  |     public ReactiveCommand<Unit, Unit> RemoveCmd { get; } | ||||||
|  |     public ReactiveCommand<Unit, Unit> MoveTopCmd { get; } | ||||||
|  |     public ReactiveCommand<Unit, Unit> MoveUpCmd { get; } | ||||||
|  |     public ReactiveCommand<Unit, Unit> MoveDownCmd { get; } | ||||||
|  |     public ReactiveCommand<Unit, Unit> MoveBottomCmd { get; } | ||||||
|  | 
 | ||||||
|  |     public ReactiveCommand<Unit, Unit> SaveCmd { get; } | ||||||
|  | 
 | ||||||
|  |     public AddGroupServerViewModel(ProfileItem profileItem, Func<EViewAction, object?, Task<bool>>? updateView) | ||||||
|  |     { | ||||||
|  |         _config = AppManager.Instance.Config; | ||||||
|  |         _updateView = updateView; | ||||||
|  | 
 | ||||||
|  |         var canEditRemove = this.WhenAnyValue( | ||||||
|  |             x => x.SelectedChild, | ||||||
|  |             SelectedChild => SelectedChild != null && !SelectedChild.Remarks.IsNullOrEmpty()); | ||||||
|  | 
 | ||||||
|  |         RemoveCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|  |         { | ||||||
|  |             await ChildRemoveAsync(); | ||||||
|  |         }, canEditRemove); | ||||||
|  |         MoveTopCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|  |         { | ||||||
|  |             await MoveServer(EMove.Top); | ||||||
|  |         }, canEditRemove); | ||||||
|  |         MoveUpCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|  |         { | ||||||
|  |             await MoveServer(EMove.Up); | ||||||
|  |         }, canEditRemove); | ||||||
|  |         MoveDownCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|  |         { | ||||||
|  |             await MoveServer(EMove.Down); | ||||||
|  |         }, canEditRemove); | ||||||
|  |         MoveBottomCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|  |         { | ||||||
|  |             await MoveServer(EMove.Bottom); | ||||||
|  |         }, canEditRemove); | ||||||
|  |         SaveCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|  |         { | ||||||
|  |             await SaveServerAsync(); | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         SelectedSource = profileItem.IndexId.IsNullOrEmpty() ? profileItem : JsonUtils.DeepCopy(profileItem); | ||||||
|  | 
 | ||||||
|  |         CoreType = (SelectedSource?.CoreType ?? ECoreType.Xray).ToString(); | ||||||
|  | 
 | ||||||
|  |         ProfileGroupItemManager.Instance.TryGet(profileItem.IndexId, out var profileGroup); | ||||||
|  |         PolicyGroupType = (profileGroup?.MultipleLoad ?? EMultipleLoad.LeastPing) switch | ||||||
|  |         { | ||||||
|  |             EMultipleLoad.LeastPing => ResUI.TbLeastPing, | ||||||
|  |             EMultipleLoad.Fallback => ResUI.TbFallback, | ||||||
|  |             EMultipleLoad.Random => ResUI.TbRandom, | ||||||
|  |             EMultipleLoad.RoundRobin => ResUI.TbRoundRobin, | ||||||
|  |             EMultipleLoad.LeastLoad => ResUI.TbLeastLoad, | ||||||
|  |             _ => ResUI.TbLeastPing, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         _ = Init(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public async Task Init() | ||||||
|  |     { | ||||||
|  |         var childItemMulti = ProfileGroupItemManager.Instance.GetOrCreateAndMarkDirty(SelectedSource?.IndexId); | ||||||
|  |         if (childItemMulti != null) | ||||||
|  |         { | ||||||
|  |             var childIndexIds = childItemMulti.ChildItems.IsNullOrEmpty() ? new List<string>() : Utils.String2List(childItemMulti.ChildItems); | ||||||
|  |             foreach (var item in childIndexIds) | ||||||
|  |             { | ||||||
|  |                 var child = await AppManager.Instance.GetProfileItem(item); | ||||||
|  |                 if (child == null) | ||||||
|  |                 { | ||||||
|  |                     continue; | ||||||
|  |                 } | ||||||
|  |                 ChildItemsObs.Add(child); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public async Task ChildRemoveAsync() | ||||||
|  |     { | ||||||
|  |         if (SelectedChild == null || SelectedChild.IndexId.IsNullOrEmpty()) | ||||||
|  |         { | ||||||
|  |             NoticeManager.Instance.Enqueue(ResUI.PleaseSelectServer); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         foreach (var it in SelectedChildren ?? [SelectedChild]) | ||||||
|  |         { | ||||||
|  |             if (it != null) | ||||||
|  |             { | ||||||
|  |                 ChildItemsObs.Remove(it); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         await Task.CompletedTask; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public async Task MoveServer(EMove eMove) | ||||||
|  |     { | ||||||
|  |         if (SelectedChild == null || SelectedChild.IndexId.IsNullOrEmpty()) | ||||||
|  |         { | ||||||
|  |             NoticeManager.Instance.Enqueue(ResUI.PleaseSelectServer); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         var index = ChildItemsObs.IndexOf(SelectedChild); | ||||||
|  |         if (index < 0) | ||||||
|  |         { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         var selectedChild = JsonUtils.DeepCopy(SelectedChild); | ||||||
|  |         switch (eMove) | ||||||
|  |         { | ||||||
|  |             case EMove.Top: | ||||||
|  |                 if (index == 0) | ||||||
|  |                 { | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |                 ChildItemsObs.RemoveAt(index); | ||||||
|  |                 ChildItemsObs.Insert(0, selectedChild); | ||||||
|  |                 break; | ||||||
|  |             case EMove.Up: | ||||||
|  |                 if (index == 0) | ||||||
|  |                 { | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |                 ChildItemsObs.RemoveAt(index); | ||||||
|  |                 ChildItemsObs.Insert(index - 1, selectedChild); | ||||||
|  |                 break; | ||||||
|  |             case EMove.Down: | ||||||
|  |                 if (index == ChildItemsObs.Count - 1) | ||||||
|  |                 { | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |                 ChildItemsObs.RemoveAt(index); | ||||||
|  |                 ChildItemsObs.Insert(index + 1, selectedChild); | ||||||
|  |                 break; | ||||||
|  |             case EMove.Bottom: | ||||||
|  |                 if (index == ChildItemsObs.Count - 1) | ||||||
|  |                 { | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |                 ChildItemsObs.RemoveAt(index); | ||||||
|  |                 ChildItemsObs.Add(selectedChild); | ||||||
|  |                 break; | ||||||
|  |             default: | ||||||
|  |                 break; | ||||||
|  |         } | ||||||
|  |         await Task.CompletedTask; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private async Task SaveServerAsync() | ||||||
|  |     { | ||||||
|  |         var remarks = SelectedSource.Remarks; | ||||||
|  |         if (remarks.IsNullOrEmpty()) | ||||||
|  |         { | ||||||
|  |             NoticeManager.Instance.Enqueue(ResUI.PleaseFillRemarks); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         if (ChildItemsObs.Count == 0) | ||||||
|  |         { | ||||||
|  |             NoticeManager.Instance.Enqueue(ResUI.PleaseAddAtLeastOneServer); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         SelectedSource.CoreType = CoreType.IsNullOrEmpty() ? ECoreType.Xray : (ECoreType)Enum.Parse(typeof(ECoreType), CoreType); | ||||||
|  |         if (SelectedSource.CoreType is not (ECoreType.Xray or ECoreType.sing_box) || | ||||||
|  |             SelectedSource.ConfigType is not (EConfigType.ProxyChain or EConfigType.PolicyGroup)) | ||||||
|  |         { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |         var childIndexIds = new List<string>(); | ||||||
|  |         foreach (var item in ChildItemsObs) | ||||||
|  |         { | ||||||
|  |             if (!item.IndexId.IsNullOrEmpty()) | ||||||
|  |             { | ||||||
|  |                 childIndexIds.Add(item.IndexId); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         SelectedSource.Address = Utils.List2String(childIndexIds); | ||||||
|  |         var profileGroup = ProfileGroupItemManager.Instance.GetOrCreateAndMarkDirty(SelectedSource.IndexId); | ||||||
|  |         profileGroup.ChildItems = Utils.List2String(childIndexIds); | ||||||
|  |         profileGroup.MultipleLoad = PolicyGroupType switch | ||||||
|  |         { | ||||||
|  |             var s when s == ResUI.TbLeastPing => EMultipleLoad.LeastPing, | ||||||
|  |             var s when s == ResUI.TbFallback => EMultipleLoad.Fallback, | ||||||
|  |             var s when s == ResUI.TbRandom => EMultipleLoad.Random, | ||||||
|  |             var s when s == ResUI.TbRoundRobin => EMultipleLoad.RoundRobin, | ||||||
|  |             var s when s == ResUI.TbLeastLoad => EMultipleLoad.LeastLoad, | ||||||
|  |             _ => EMultipleLoad.LeastPing, | ||||||
|  |         }; | ||||||
|  |         if (await ConfigHandler.AddGroupServerCommon(_config, SelectedSource, profileGroup, true) == 0) | ||||||
|  |         { | ||||||
|  |             NoticeManager.Instance.Enqueue(ResUI.OperationSuccess); | ||||||
|  |             _updateView?.Invoke(EViewAction.CloseWindow, null); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             NoticeManager.Instance.Enqueue(ResUI.OperationFailed); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -23,6 +23,8 @@ public class MainWindowViewModel : MyReactiveObject | ||||||
|     public ReactiveCommand<Unit, Unit> AddWireguardServerCmd { get; } |     public ReactiveCommand<Unit, Unit> AddWireguardServerCmd { get; } | ||||||
|     public ReactiveCommand<Unit, Unit> AddAnytlsServerCmd { get; } |     public ReactiveCommand<Unit, Unit> AddAnytlsServerCmd { get; } | ||||||
|     public ReactiveCommand<Unit, Unit> AddCustomServerCmd { get; } |     public ReactiveCommand<Unit, Unit> AddCustomServerCmd { get; } | ||||||
|  |     public ReactiveCommand<Unit, Unit> AddPolicyGroupServerCmd { get; } | ||||||
|  |     public ReactiveCommand<Unit, Unit> AddProxyChainServerCmd { get; } | ||||||
|     public ReactiveCommand<Unit, Unit> AddServerViaClipboardCmd { get; } |     public ReactiveCommand<Unit, Unit> AddServerViaClipboardCmd { get; } | ||||||
|     public ReactiveCommand<Unit, Unit> AddServerViaScanCmd { get; } |     public ReactiveCommand<Unit, Unit> AddServerViaScanCmd { get; } | ||||||
|     public ReactiveCommand<Unit, Unit> AddServerViaImageCmd { get; } |     public ReactiveCommand<Unit, Unit> AddServerViaImageCmd { get; } | ||||||
|  | @ -122,6 +124,14 @@ public class MainWindowViewModel : MyReactiveObject | ||||||
|         { |         { | ||||||
|             await AddServerAsync(true, EConfigType.Custom); |             await AddServerAsync(true, EConfigType.Custom); | ||||||
|         }); |         }); | ||||||
|  |         AddPolicyGroupServerCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|  |         { | ||||||
|  |             await AddServerAsync(true, EConfigType.PolicyGroup); | ||||||
|  |         }); | ||||||
|  |         AddProxyChainServerCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|  |         { | ||||||
|  |             await AddServerAsync(true, EConfigType.ProxyChain); | ||||||
|  |         }); | ||||||
|         AddServerViaClipboardCmd = ReactiveCommand.CreateFromTask(async () => |         AddServerViaClipboardCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|         { |         { | ||||||
|             await AddServerViaClipboardAsync(null); |             await AddServerViaClipboardAsync(null); | ||||||
|  | @ -252,6 +262,7 @@ public class MainWindowViewModel : MyReactiveObject | ||||||
|         await ConfigHandler.InitBuiltinDNS(_config); |         await ConfigHandler.InitBuiltinDNS(_config); | ||||||
|         await ConfigHandler.InitBuiltinFullConfigTemplate(_config); |         await ConfigHandler.InitBuiltinFullConfigTemplate(_config); | ||||||
|         await ProfileExManager.Instance.Init(); |         await ProfileExManager.Instance.Init(); | ||||||
|  |         await ProfileGroupItemManager.Instance.Init(); | ||||||
|         await CoreManager.Instance.Init(_config, UpdateHandler); |         await CoreManager.Instance.Init(_config, UpdateHandler); | ||||||
|         TaskManager.Instance.RegUpdateTask(_config, UpdateTaskHandler); |         TaskManager.Instance.RegUpdateTask(_config, UpdateTaskHandler); | ||||||
| 
 | 
 | ||||||
|  | @ -340,6 +351,10 @@ public class MainWindowViewModel : MyReactiveObject | ||||||
|         { |         { | ||||||
|             ret = await _updateView?.Invoke(EViewAction.AddServer2Window, item); |             ret = await _updateView?.Invoke(EViewAction.AddServer2Window, item); | ||||||
|         } |         } | ||||||
|  |         else if (eConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) | ||||||
|  |         { | ||||||
|  |             ret = await _updateView?.Invoke(EViewAction.AddGroupServerWindow, item); | ||||||
|  |         } | ||||||
|         else |         else | ||||||
|         { |         { | ||||||
|             ret = await _updateView?.Invoke(EViewAction.AddServerWindow, item); |             ret = await _updateView?.Invoke(EViewAction.AddServerWindow, item); | ||||||
|  |  | ||||||
|  | @ -1,5 +1,6 @@ | ||||||
| using System.Collections.Concurrent; | using System.Collections.Concurrent; | ||||||
| using System.Reactive.Linq; | using System.Reactive.Linq; | ||||||
|  | using System.Text; | ||||||
| using System.Text.RegularExpressions; | using System.Text.RegularExpressions; | ||||||
| using ReactiveUI; | using ReactiveUI; | ||||||
| using ReactiveUI.Fody.Helpers; | using ReactiveUI.Fody.Helpers; | ||||||
|  | @ -9,9 +10,9 @@ namespace ServiceLib.ViewModels; | ||||||
| public class MsgViewModel : MyReactiveObject | public class MsgViewModel : MyReactiveObject | ||||||
| { | { | ||||||
|     private readonly ConcurrentQueue<string> _queueMsg = new(); |     private readonly ConcurrentQueue<string> _queueMsg = new(); | ||||||
|     private readonly int _numMaxMsg = 500; |     private volatile bool _lastMsgFilterNotAvailable; | ||||||
|     private bool _lastMsgFilterNotAvailable; |     private int _showLock = 0; // 0 = unlocked, 1 = locked | ||||||
|     private bool _blLockShow = false; |     public int NumMaxMsg { get; } = 500; | ||||||
| 
 | 
 | ||||||
|     [Reactive] |     [Reactive] | ||||||
|     public string MsgFilter { get; set; } |     public string MsgFilter { get; set; } | ||||||
|  | @ -33,46 +34,52 @@ public class MsgViewModel : MyReactiveObject | ||||||
|         this.WhenAnyValue( |         this.WhenAnyValue( | ||||||
|           x => x.AutoRefresh, |           x => x.AutoRefresh, | ||||||
|           y => y == true) |           y => y == true) | ||||||
|               .Subscribe(c => { _config.MsgUIItem.AutoRefresh = AutoRefresh; }); |               .Subscribe(c => _config.MsgUIItem.AutoRefresh = AutoRefresh); | ||||||
| 
 | 
 | ||||||
|         AppEvents.SendMsgViewRequested |         AppEvents.SendMsgViewRequested | ||||||
|          .AsObservable() |          .AsObservable() | ||||||
|          //.ObserveOn(RxApp.MainThreadScheduler) |          //.ObserveOn(RxApp.MainThreadScheduler) | ||||||
|          .Subscribe(async content => await AppendQueueMsg(content)); |          .Subscribe(content => _ = AppendQueueMsg(content)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private async Task AppendQueueMsg(string msg) |     private async Task AppendQueueMsg(string msg) | ||||||
|     { |     { | ||||||
|         //if (msg == Global.CommandClearMsg) |  | ||||||
|         //{ |  | ||||||
|         //    ClearMsg(); |  | ||||||
|         //    return; |  | ||||||
|         //} |  | ||||||
|         if (AutoRefresh == false) |         if (AutoRefresh == false) | ||||||
|         { |         { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|         _ = EnqueueQueueMsg(msg); |  | ||||||
| 
 | 
 | ||||||
|         if (_blLockShow) |         EnqueueQueueMsg(msg); | ||||||
|         { | 
 | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|         if (!_config.UiItem.ShowInTaskbar) |         if (!_config.UiItem.ShowInTaskbar) | ||||||
|         { |         { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         _blLockShow = true; |         if (Interlocked.CompareExchange(ref _showLock, 1, 0) != 0) | ||||||
|  |         { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         await Task.Delay(500); |         try | ||||||
|         var txt = string.Join("", _queueMsg.ToArray()); |         { | ||||||
|         await _updateView?.Invoke(EViewAction.DispatcherShowMsg, txt); |             await Task.Delay(500).ConfigureAwait(false); | ||||||
| 
 | 
 | ||||||
|         _blLockShow = false; |             var sb = new StringBuilder(); | ||||||
|  |             while (_queueMsg.TryDequeue(out var line)) | ||||||
|  |             { | ||||||
|  |                 sb.Append(line); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             await _updateView?.Invoke(EViewAction.DispatcherShowMsg, sb.ToString()); | ||||||
|  |         } | ||||||
|  |         finally | ||||||
|  |         { | ||||||
|  |             Interlocked.Exchange(ref _showLock, 0); | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private async Task EnqueueQueueMsg(string msg) |     private void EnqueueQueueMsg(string msg) | ||||||
|     { |     { | ||||||
|         //filter msg |         //filter msg | ||||||
|         if (MsgFilter.IsNotEmpty() && !_lastMsgFilterNotAvailable) |         if (MsgFilter.IsNotEmpty() && !_lastMsgFilterNotAvailable) | ||||||
|  | @ -91,26 +98,17 @@ public class MsgViewModel : MyReactiveObject | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         //Enqueue |  | ||||||
|         if (_queueMsg.Count > _numMaxMsg) |  | ||||||
|         { |  | ||||||
|             for (int k = 0; k < _queueMsg.Count - _numMaxMsg; k++) |  | ||||||
|             { |  | ||||||
|                 _queueMsg.TryDequeue(out _); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         _queueMsg.Enqueue(msg); |         _queueMsg.Enqueue(msg); | ||||||
|         if (!msg.EndsWith(Environment.NewLine)) |         if (!msg.EndsWith(Environment.NewLine)) | ||||||
|         { |         { | ||||||
|             _queueMsg.Enqueue(Environment.NewLine); |             _queueMsg.Enqueue(Environment.NewLine); | ||||||
|         } |         } | ||||||
|         await Task.CompletedTask; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public void ClearMsg() |     //public void ClearMsg() | ||||||
|     { |     //{ | ||||||
|         _queueMsg.Clear(); |     //    _queueMsg.Clear(); | ||||||
|     } |     //} | ||||||
| 
 | 
 | ||||||
|     private void DoMsgFilter() |     private void DoMsgFilter() | ||||||
|     { |     { | ||||||
|  |  | ||||||
|  | @ -52,11 +52,13 @@ public class ProfilesViewModel : MyReactiveObject | ||||||
|     public ReactiveCommand<Unit, Unit> CopyServerCmd { get; } |     public ReactiveCommand<Unit, Unit> CopyServerCmd { get; } | ||||||
|     public ReactiveCommand<Unit, Unit> SetDefaultServerCmd { get; } |     public ReactiveCommand<Unit, Unit> SetDefaultServerCmd { get; } | ||||||
|     public ReactiveCommand<Unit, Unit> ShareServerCmd { get; } |     public ReactiveCommand<Unit, Unit> ShareServerCmd { get; } | ||||||
|     public ReactiveCommand<Unit, Unit> SetDefaultMultipleServerXrayRandomCmd { get; } |     public ReactiveCommand<Unit, Unit> GenGroupMultipleServerXrayRandomCmd { get; } | ||||||
|     public ReactiveCommand<Unit, Unit> SetDefaultMultipleServerXrayRoundRobinCmd { get; } |     public ReactiveCommand<Unit, Unit> GenGroupMultipleServerXrayRoundRobinCmd { get; } | ||||||
|     public ReactiveCommand<Unit, Unit> SetDefaultMultipleServerXrayLeastPingCmd { get; } |     public ReactiveCommand<Unit, Unit> GenGroupMultipleServerXrayLeastPingCmd { get; } | ||||||
|     public ReactiveCommand<Unit, Unit> SetDefaultMultipleServerXrayLeastLoadCmd { get; } |     public ReactiveCommand<Unit, Unit> GenGroupMultipleServerXrayLeastLoadCmd { get; } | ||||||
|     public ReactiveCommand<Unit, Unit> SetDefaultMultipleServerSingBoxLeastPingCmd { get; } |     public ReactiveCommand<Unit, Unit> GenGroupMultipleServerXrayFallbackCmd { get; } | ||||||
|  |     public ReactiveCommand<Unit, Unit> GenGroupMultipleServerSingBoxLeastPingCmd { get; } | ||||||
|  |     public ReactiveCommand<Unit, Unit> GenGroupMultipleServerSingBoxFallbackCmd { get; } | ||||||
| 
 | 
 | ||||||
|     //servers move |     //servers move | ||||||
|     public ReactiveCommand<Unit, Unit> MoveTopCmd { get; } |     public ReactiveCommand<Unit, Unit> MoveTopCmd { get; } | ||||||
|  | @ -138,25 +140,33 @@ public class ProfilesViewModel : MyReactiveObject | ||||||
|         { |         { | ||||||
|             await ShareServerAsync(); |             await ShareServerAsync(); | ||||||
|         }, canEditRemove); |         }, canEditRemove); | ||||||
|         SetDefaultMultipleServerXrayRandomCmd = ReactiveCommand.CreateFromTask(async () => |         GenGroupMultipleServerXrayRandomCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|         { |         { | ||||||
|             await SetDefaultMultipleServer(ECoreType.Xray, EMultipleLoad.Random); |             await GenGroupMultipleServer(ECoreType.Xray, EMultipleLoad.Random); | ||||||
|         }, canEditRemove); |         }, canEditRemove); | ||||||
|         SetDefaultMultipleServerXrayRoundRobinCmd = ReactiveCommand.CreateFromTask(async () => |         GenGroupMultipleServerXrayRoundRobinCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|         { |         { | ||||||
|             await SetDefaultMultipleServer(ECoreType.Xray, EMultipleLoad.RoundRobin); |             await GenGroupMultipleServer(ECoreType.Xray, EMultipleLoad.RoundRobin); | ||||||
|         }, canEditRemove); |         }, canEditRemove); | ||||||
|         SetDefaultMultipleServerXrayLeastPingCmd = ReactiveCommand.CreateFromTask(async () => |         GenGroupMultipleServerXrayLeastPingCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|         { |         { | ||||||
|             await SetDefaultMultipleServer(ECoreType.Xray, EMultipleLoad.LeastPing); |             await GenGroupMultipleServer(ECoreType.Xray, EMultipleLoad.LeastPing); | ||||||
|         }, canEditRemove); |         }, canEditRemove); | ||||||
|         SetDefaultMultipleServerXrayLeastLoadCmd = ReactiveCommand.CreateFromTask(async () => |         GenGroupMultipleServerXrayLeastLoadCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|         { |         { | ||||||
|             await SetDefaultMultipleServer(ECoreType.Xray, EMultipleLoad.LeastLoad); |             await GenGroupMultipleServer(ECoreType.Xray, EMultipleLoad.LeastLoad); | ||||||
|         }, canEditRemove); |         }, canEditRemove); | ||||||
|         SetDefaultMultipleServerSingBoxLeastPingCmd = ReactiveCommand.CreateFromTask(async () => |         GenGroupMultipleServerXrayFallbackCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|         { |         { | ||||||
|             await SetDefaultMultipleServer(ECoreType.sing_box, EMultipleLoad.LeastPing); |             await GenGroupMultipleServer(ECoreType.Xray, EMultipleLoad.Fallback); | ||||||
|  |         }, canEditRemove); | ||||||
|  |         GenGroupMultipleServerSingBoxLeastPingCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|  |         { | ||||||
|  |             await GenGroupMultipleServer(ECoreType.sing_box, EMultipleLoad.LeastPing); | ||||||
|  |         }, canEditRemove); | ||||||
|  |         GenGroupMultipleServerSingBoxFallbackCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|  |         { | ||||||
|  |             await GenGroupMultipleServer(ECoreType.sing_box, EMultipleLoad.Fallback); | ||||||
|         }, canEditRemove); |         }, canEditRemove); | ||||||
| 
 | 
 | ||||||
|         //servers move |         //servers move | ||||||
|  | @ -500,6 +510,10 @@ public class ProfilesViewModel : MyReactiveObject | ||||||
|         { |         { | ||||||
|             ret = await _updateView?.Invoke(EViewAction.AddServer2Window, item); |             ret = await _updateView?.Invoke(EViewAction.AddServer2Window, item); | ||||||
|         } |         } | ||||||
|  |         else if (eConfigType is EConfigType.PolicyGroup or EConfigType.ProxyChain) | ||||||
|  |         { | ||||||
|  |             ret = await _updateView?.Invoke(EViewAction.AddGroupServerWindow, item); | ||||||
|  |         } | ||||||
|         else |         else | ||||||
|         { |         { | ||||||
|             ret = await _updateView?.Invoke(EViewAction.AddServerWindow, item); |             ret = await _updateView?.Invoke(EViewAction.AddServerWindow, item); | ||||||
|  | @ -615,7 +629,7 @@ public class ProfilesViewModel : MyReactiveObject | ||||||
|         await _updateView?.Invoke(EViewAction.ShareServer, url); |         await _updateView?.Invoke(EViewAction.ShareServer, url); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private async Task SetDefaultMultipleServer(ECoreType coreType, EMultipleLoad multipleLoad) |     private async Task GenGroupMultipleServer(ECoreType coreType, EMultipleLoad multipleLoad) | ||||||
|     { |     { | ||||||
|         var lstSelected = await GetProfileItems(true); |         var lstSelected = await GetProfileItems(true); | ||||||
|         if (lstSelected == null) |         if (lstSelected == null) | ||||||
|  | @ -623,7 +637,7 @@ public class ProfilesViewModel : MyReactiveObject | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         var ret = await ConfigHandler.AddCustomServer4Multiple(_config, lstSelected, coreType, multipleLoad); |         var ret = await ConfigHandler.AddGroupServer4Multiple(_config, lstSelected, coreType, multipleLoad, SelectedSub?.Id); | ||||||
|         if (ret.Success != true) |         if (ret.Success != true) | ||||||
|         { |         { | ||||||
|             NoticeManager.Instance.Enqueue(ResUI.OperationFailed); |             NoticeManager.Instance.Enqueue(ResUI.OperationFailed); | ||||||
|  |  | ||||||
|  | @ -11,6 +11,7 @@ | ||||||
|     RequestedThemeVariant="Default"> |     RequestedThemeVariant="Default"> | ||||||
|     <Application.Styles> |     <Application.Styles> | ||||||
|         <semi:SemiTheme /> |         <semi:SemiTheme /> | ||||||
|  |         <semi:AvaloniaEditSemiTheme /> | ||||||
|         <StyleInclude Source="Assets/GlobalStyles.axaml" /> |         <StyleInclude Source="Assets/GlobalStyles.axaml" /> | ||||||
|         <StyleInclude Source="avares://Semi.Avalonia.DataGrid/Index.axaml" /> |         <StyleInclude Source="avares://Semi.Avalonia.DataGrid/Index.axaml" /> | ||||||
|         <dialogHost:DialogHostStyles /> |         <dialogHost:DialogHostStyles /> | ||||||
|  |  | ||||||
							
								
								
									
										129
									
								
								v2rayN/v2rayN.Desktop/Common/TextEditorKeywordHighlighter.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										129
									
								
								v2rayN/v2rayN.Desktop/Common/TextEditorKeywordHighlighter.cs
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,129 @@ | ||||||
|  | using Avalonia.Media; | ||||||
|  | using AvaloniaEdit; | ||||||
|  | using AvaloniaEdit.Document; | ||||||
|  | using AvaloniaEdit.Rendering; | ||||||
|  | 
 | ||||||
|  | namespace v2rayN.Desktop.Common; | ||||||
|  | 
 | ||||||
|  | public class KeywordColorizer : DocumentColorizingTransformer | ||||||
|  | { | ||||||
|  |     private readonly string[] _keywords; | ||||||
|  |     private readonly Dictionary<string, IBrush> _brushMap; | ||||||
|  | 
 | ||||||
|  |     public KeywordColorizer(IDictionary<string, IBrush> keywordBrushMap) | ||||||
|  |     { | ||||||
|  |         if (keywordBrushMap == null || keywordBrushMap.Count == 0) | ||||||
|  |         { | ||||||
|  |             throw new ArgumentException("keywordBrushMap must not be null or empty", nameof(keywordBrushMap)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         _brushMap = new Dictionary<string, IBrush>(StringComparer.OrdinalIgnoreCase); | ||||||
|  |         foreach (var kvp in keywordBrushMap) | ||||||
|  |         { | ||||||
|  |             if (string.IsNullOrEmpty(kvp.Key) || kvp.Value == null) | ||||||
|  |             { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (!_brushMap.ContainsKey(kvp.Key)) | ||||||
|  |             { | ||||||
|  |                 _brushMap[kvp.Key] = kvp.Value; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (_brushMap.Count == 0) | ||||||
|  |         { | ||||||
|  |             throw new ArgumentException("keywordBrushMap must contain at least one non-empty key with a non-null brush", nameof(keywordBrushMap)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         _keywords = _brushMap.Keys.ToArray(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     protected override void ColorizeLine(DocumentLine line) | ||||||
|  |     { | ||||||
|  |         var text = CurrentContext.Document.GetText(line); | ||||||
|  |         if (string.IsNullOrEmpty(text)) | ||||||
|  |         { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         foreach (var kw in _keywords) | ||||||
|  |         { | ||||||
|  |             if (string.IsNullOrEmpty(kw)) | ||||||
|  |             { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var searchStart = 0; | ||||||
|  |             while (true) | ||||||
|  |             { | ||||||
|  |                 var idx = text.IndexOf(kw, searchStart, StringComparison.OrdinalIgnoreCase); | ||||||
|  |                 if (idx < 0) | ||||||
|  |                 { | ||||||
|  |                     break; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 var kwEndIndex = idx + kw.Length; | ||||||
|  |                 if (IsWordCharBefore(text, idx) || IsWordCharAfter(text, kwEndIndex)) | ||||||
|  |                 { | ||||||
|  |                     searchStart = idx + Math.Max(1, kw.Length); | ||||||
|  |                     continue; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 var start = line.Offset + idx; | ||||||
|  |                 var end = start + kw.Length; | ||||||
|  | 
 | ||||||
|  |                 if (_brushMap.TryGetValue(kw, out var brush) && brush != null) | ||||||
|  |                 { | ||||||
|  |                     ChangeLinePart(start, end, element => element.TextRunProperties.SetForegroundBrush(brush)); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 searchStart = idx + Math.Max(1, kw.Length); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static bool IsWordCharBefore(string text, int idx) | ||||||
|  |     { | ||||||
|  |         if (idx <= 0) | ||||||
|  |         { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         var c = text[idx - 1]; | ||||||
|  |         return char.IsLetterOrDigit(c) || c == '_'; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static bool IsWordCharAfter(string text, int idx) | ||||||
|  |     { | ||||||
|  |         if (idx >= text.Length) | ||||||
|  |         { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         var c = text[idx]; | ||||||
|  |         return char.IsLetterOrDigit(c) || c == '_'; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | public static class TextEditorKeywordHighlighter | ||||||
|  | { | ||||||
|  |     public static void Attach(TextEditor editor, IDictionary<string, IBrush> keywordBrushMap) | ||||||
|  |     { | ||||||
|  |         ArgumentNullException.ThrowIfNull(editor); | ||||||
|  | 
 | ||||||
|  |         if (keywordBrushMap == null || keywordBrushMap.Count == 0) | ||||||
|  |         { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (editor.TextArea?.TextView?.LineTransformers?.OfType<KeywordColorizer>().Any() == true) | ||||||
|  |         { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         var colorizer = new KeywordColorizer(keywordBrushMap); | ||||||
|  |         editor.TextArea.TextView.LineTransformers.Add(colorizer); | ||||||
|  |         editor.TextArea.TextView.InvalidateVisual(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -5,6 +5,7 @@ using Avalonia.Controls.Notifications; | ||||||
| using Avalonia.Controls.Primitives; | using Avalonia.Controls.Primitives; | ||||||
| using Avalonia.Media; | using Avalonia.Media; | ||||||
| using Avalonia.Styling; | using Avalonia.Styling; | ||||||
|  | using AvaloniaEdit; | ||||||
| using ReactiveUI; | using ReactiveUI; | ||||||
| using ReactiveUI.Fody.Helpers; | using ReactiveUI.Fody.Helpers; | ||||||
| using Semi.Avalonia; | using Semi.Avalonia; | ||||||
|  | @ -112,7 +113,8 @@ public class ThemeSettingViewModel : MyReactiveObject | ||||||
|             x.OfType<ContextMenu>(), |             x.OfType<ContextMenu>(), | ||||||
|             x.OfType<DataGridRow>(), |             x.OfType<DataGridRow>(), | ||||||
|             x.OfType<ListBoxItem>(), |             x.OfType<ListBoxItem>(), | ||||||
|             x.OfType<HeaderedContentControl>() |             x.OfType<HeaderedContentControl>(), | ||||||
|  |             x.OfType<TextEditor>() | ||||||
|         )); |         )); | ||||||
|         style.Add(new Setter() |         style.Add(new Setter() | ||||||
|         { |         { | ||||||
|  | @ -153,7 +155,8 @@ public class ThemeSettingViewModel : MyReactiveObject | ||||||
|                 x.OfType<DataGridRow>(), |                 x.OfType<DataGridRow>(), | ||||||
|                 x.OfType<ListBoxItem>(), |                 x.OfType<ListBoxItem>(), | ||||||
|                 x.OfType<HeaderedContentControl>(), |                 x.OfType<HeaderedContentControl>(), | ||||||
|                 x.OfType<WindowNotificationManager>() |                 x.OfType<WindowNotificationManager>(), | ||||||
|  |                 x.OfType<TextEditor>() | ||||||
|             )); |             )); | ||||||
|             style.Add(new Setter() |             style.Add(new Setter() | ||||||
|             { |             { | ||||||
|  |  | ||||||
							
								
								
									
										151
									
								
								v2rayN/v2rayN.Desktop/Views/AddGroupServerWindow.axaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										151
									
								
								v2rayN/v2rayN.Desktop/Views/AddGroupServerWindow.axaml
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,151 @@ | ||||||
|  | <Window | ||||||
|  |     x:Class="v2rayN.Desktop.Views.AddGroupServerWindow" | ||||||
|  |     xmlns="https://github.com/avaloniaui" | ||||||
|  |     xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" | ||||||
|  |     xmlns:d="http://schemas.microsoft.com/expression/blend/2008" | ||||||
|  |     xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" | ||||||
|  |     xmlns:resx="clr-namespace:ServiceLib.Resx;assembly=ServiceLib" | ||||||
|  |     xmlns:vms="clr-namespace:ServiceLib.ViewModels;assembly=ServiceLib" | ||||||
|  |     Title="{x:Static resx:ResUI.menuServers}" | ||||||
|  |     Width="900" | ||||||
|  |     Height="700" | ||||||
|  |     x:DataType="vms:AddGroupServerViewModel" | ||||||
|  |     ShowInTaskbar="False" | ||||||
|  |     WindowStartupLocation="CenterScreen" | ||||||
|  |     mc:Ignorable="d"> | ||||||
|  | 
 | ||||||
|  |     <DockPanel Margin="{StaticResource Margin8}"> | ||||||
|  |         <StackPanel | ||||||
|  |             Margin="{StaticResource Margin4}" | ||||||
|  |             HorizontalAlignment="Center" | ||||||
|  |             DockPanel.Dock="Bottom" | ||||||
|  |             Orientation="Horizontal"> | ||||||
|  |             <Button | ||||||
|  |                 x:Name="btnSave" | ||||||
|  |                 Width="100" | ||||||
|  |                 Content="{x:Static resx:ResUI.TbConfirm}" | ||||||
|  |                 IsDefault="True" /> | ||||||
|  |             <Button | ||||||
|  |                 x:Name="btnCancel" | ||||||
|  |                 Width="100" | ||||||
|  |                 Margin="{StaticResource MarginLr8}" | ||||||
|  |                 Content="{x:Static resx:ResUI.TbCancel}" | ||||||
|  |                 IsCancel="True" /> | ||||||
|  |         </StackPanel> | ||||||
|  | 
 | ||||||
|  |         <Grid DockPanel.Dock="Top" RowDefinitions="Auto,*,Auto,Auto,Auto,Auto,Auto,Auto,Auto"> | ||||||
|  | 
 | ||||||
|  |             <Grid | ||||||
|  |                 Grid.Row="0" | ||||||
|  |                 ColumnDefinitions="180,Auto,Auto" | ||||||
|  |                 RowDefinitions="Auto,Auto,Auto,Auto,Auto"> | ||||||
|  |                 <TextBlock | ||||||
|  |                     Grid.Row="0" | ||||||
|  |                     Grid.Column="0" | ||||||
|  |                     Margin="{StaticResource Margin4}" | ||||||
|  |                     Text="{x:Static resx:ResUI.menuServers}" /> | ||||||
|  | 
 | ||||||
|  |                 <TextBlock | ||||||
|  |                     Grid.Row="1" | ||||||
|  |                     Grid.Column="0" | ||||||
|  |                     Margin="{StaticResource Margin4}" | ||||||
|  |                     VerticalAlignment="Center" | ||||||
|  |                     Text="{x:Static resx:ResUI.TbRemarks}" /> | ||||||
|  |                 <TextBox | ||||||
|  |                     x:Name="txtRemarks" | ||||||
|  |                     Grid.Row="1" | ||||||
|  |                     Grid.Column="1" | ||||||
|  |                     Width="400" | ||||||
|  |                     Margin="{StaticResource Margin4}" /> | ||||||
|  | 
 | ||||||
|  |                 <TextBlock | ||||||
|  |                     Grid.Row="2" | ||||||
|  |                     Grid.Column="0" | ||||||
|  |                     Margin="{StaticResource Margin4}" | ||||||
|  |                     VerticalAlignment="Center" | ||||||
|  |                     Text="{x:Static resx:ResUI.TbCoreType}" /> | ||||||
|  |                 <ComboBox | ||||||
|  |                     x:Name="cmbCoreType" | ||||||
|  |                     Grid.Row="2" | ||||||
|  |                     Grid.Column="1" | ||||||
|  |                     Width="200" | ||||||
|  |                     Margin="{StaticResource Margin4}" /> | ||||||
|  | 
 | ||||||
|  |                 <Grid | ||||||
|  |                     x:Name="gridPolicyGroup" | ||||||
|  |                     Grid.Row="3" | ||||||
|  |                     Grid.Column="0" | ||||||
|  |                     Grid.ColumnSpan="3" | ||||||
|  |                     ColumnDefinitions="180,Auto,Auto"> | ||||||
|  |                     <TextBlock | ||||||
|  |                         Grid.Column="0" | ||||||
|  |                         Margin="{StaticResource Margin4}" | ||||||
|  |                         VerticalAlignment="Center" | ||||||
|  |                         Text="{x:Static resx:ResUI.TbPolicyGroupType}" /> | ||||||
|  |                     <ComboBox | ||||||
|  |                         x:Name="cmbPolicyGroupType" | ||||||
|  |                         Grid.Column="1" | ||||||
|  |                         Width="200" | ||||||
|  |                         Margin="{StaticResource Margin4}" /> | ||||||
|  |                 </Grid> | ||||||
|  |             </Grid> | ||||||
|  |         </Grid> | ||||||
|  |         <TabControl> | ||||||
|  |             <TabItem HorizontalAlignment="Left" Header="{x:Static resx:ResUI.menuServerList}"> | ||||||
|  |                 <DataGrid | ||||||
|  |                     x:Name="lstChild" | ||||||
|  |                     Grid.Row="1" | ||||||
|  |                     AutoGenerateColumns="False" | ||||||
|  |                     Background="Transparent" | ||||||
|  |                     BorderThickness="1" | ||||||
|  |                     CanUserReorderColumns="False" | ||||||
|  |                     CanUserResizeColumns="True" | ||||||
|  |                     CanUserSortColumns="False" | ||||||
|  |                     GridLinesVisibility="All" | ||||||
|  |                     HeadersVisibility="Column" | ||||||
|  |                     IsReadOnly="True" | ||||||
|  |                     ItemsSource="{Binding ChildItemsObs}" | ||||||
|  |                     SelectionMode="Extended"> | ||||||
|  |                     <DataGrid.ContextMenu> | ||||||
|  |                         <ContextMenu> | ||||||
|  |                             <MenuItem x:Name="menuAddChildServer" Header="{x:Static resx:ResUI.menuAddChildServer}" /> | ||||||
|  |                             <MenuItem x:Name="menuRemoveChildServer" Header="{x:Static resx:ResUI.menuRemoveChildServer}" /> | ||||||
|  |                             <MenuItem x:Name="menuSelectAllChild" Header="{x:Static resx:ResUI.menuSelectAll}" /> | ||||||
|  |                             <Separator /> | ||||||
|  |                             <MenuItem x:Name="menuMoveTop" Header="{x:Static resx:ResUI.menuMoveTop}" /> | ||||||
|  |                             <MenuItem x:Name="menuMoveUp" Header="{x:Static resx:ResUI.menuMoveUp}" /> | ||||||
|  |                             <MenuItem x:Name="menuMoveDown" Header="{x:Static resx:ResUI.menuMoveDown}" /> | ||||||
|  |                             <MenuItem x:Name="menuMoveBottom" Header="{x:Static resx:ResUI.menuMoveBottom}" /> | ||||||
|  |                         </ContextMenu> | ||||||
|  |                     </DataGrid.ContextMenu> | ||||||
|  |                     <DataGrid.Columns> | ||||||
|  |                         <DataGridTextColumn | ||||||
|  |                             Width="150" | ||||||
|  |                             Binding="{Binding ConfigType}" | ||||||
|  |                             Header="{x:Static resx:ResUI.LvServiceType}" /> | ||||||
|  |                         <DataGridTextColumn | ||||||
|  |                             Width="150" | ||||||
|  |                             Binding="{Binding Remarks}" | ||||||
|  |                             Header="{x:Static resx:ResUI.LvRemarks}" /> | ||||||
|  |                         <DataGridTextColumn | ||||||
|  |                             Width="120" | ||||||
|  |                             Binding="{Binding Address}" | ||||||
|  |                             Header="{x:Static resx:ResUI.LvAddress}" /> | ||||||
|  |                         <DataGridTextColumn | ||||||
|  |                             Width="100" | ||||||
|  |                             Binding="{Binding Port}" | ||||||
|  |                             Header="{x:Static resx:ResUI.LvPort}" /> | ||||||
|  |                         <DataGridTextColumn | ||||||
|  |                             Width="100" | ||||||
|  |                             Binding="{Binding Network}" | ||||||
|  |                             Header="{x:Static resx:ResUI.LvTransportProtocol}" /> | ||||||
|  |                         <DataGridTextColumn | ||||||
|  |                             Width="100" | ||||||
|  |                             Binding="{Binding StreamSecurity}" | ||||||
|  |                             Header="{x:Static resx:ResUI.LvTLS}" /> | ||||||
|  |                     </DataGrid.Columns> | ||||||
|  |                 </DataGrid> | ||||||
|  |             </TabItem> | ||||||
|  |         </TabControl> | ||||||
|  |     </DockPanel> | ||||||
|  | </Window> | ||||||
							
								
								
									
										166
									
								
								v2rayN/v2rayN.Desktop/Views/AddGroupServerWindow.axaml.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										166
									
								
								v2rayN/v2rayN.Desktop/Views/AddGroupServerWindow.axaml.cs
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,166 @@ | ||||||
|  | using System.Reactive.Disposables; | ||||||
|  | using Avalonia.Controls; | ||||||
|  | using Avalonia.Input; | ||||||
|  | using Avalonia.Interactivity; | ||||||
|  | using DynamicData; | ||||||
|  | using ReactiveUI; | ||||||
|  | using v2rayN.Desktop.Base; | ||||||
|  | 
 | ||||||
|  | namespace v2rayN.Desktop.Views; | ||||||
|  | 
 | ||||||
|  | public partial class AddGroupServerWindow : WindowBase<AddGroupServerViewModel> | ||||||
|  | { | ||||||
|  |     public AddGroupServerWindow() | ||||||
|  |     { | ||||||
|  |         InitializeComponent(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public AddGroupServerWindow(ProfileItem profileItem) | ||||||
|  |     { | ||||||
|  |         InitializeComponent(); | ||||||
|  | 
 | ||||||
|  |         this.Loaded += Window_Loaded; | ||||||
|  |         btnCancel.Click += (s, e) => this.Close(); | ||||||
|  |         lstChild.SelectionChanged += LstChild_SelectionChanged; | ||||||
|  | 
 | ||||||
|  |         ViewModel = new AddGroupServerViewModel(profileItem, UpdateViewHandler); | ||||||
|  | 
 | ||||||
|  |         cmbCoreType.ItemsSource = Global.CoreTypes; | ||||||
|  |         cmbPolicyGroupType.ItemsSource = new List<string> | ||||||
|  |         { | ||||||
|  |             ResUI.TbLeastPing, | ||||||
|  |             ResUI.TbFallback, | ||||||
|  |             ResUI.TbRandom, | ||||||
|  |             ResUI.TbRoundRobin, | ||||||
|  |             ResUI.TbLeastLoad, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         switch (profileItem.ConfigType) | ||||||
|  |         { | ||||||
|  |             case EConfigType.PolicyGroup: | ||||||
|  |                 this.Title = ResUI.TbConfigTypePolicyGroup; | ||||||
|  |                 break; | ||||||
|  |             case EConfigType.ProxyChain: | ||||||
|  |                 this.Title = ResUI.TbConfigTypeProxyChain; | ||||||
|  |                 gridPolicyGroup.IsVisible = false; | ||||||
|  |                 break; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         this.WhenActivated(disposables => | ||||||
|  |         { | ||||||
|  |             this.Bind(ViewModel, vm => vm.SelectedSource.Remarks, v => v.txtRemarks.Text).DisposeWith(disposables); | ||||||
|  |             this.Bind(ViewModel, vm => vm.CoreType, v => v.cmbCoreType.SelectedValue).DisposeWith(disposables); | ||||||
|  |             this.Bind(ViewModel, vm => vm.PolicyGroupType, v => v.cmbPolicyGroupType.SelectedValue).DisposeWith(disposables); | ||||||
|  | 
 | ||||||
|  |             this.OneWayBind(ViewModel, vm => vm.ChildItemsObs, v => v.lstChild.ItemsSource).DisposeWith(disposables); | ||||||
|  |             this.Bind(ViewModel, vm => vm.SelectedChild, v => v.lstChild.SelectedItem).DisposeWith(disposables); | ||||||
|  | 
 | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.RemoveCmd, v => v.menuRemoveChildServer).DisposeWith(disposables); | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.MoveTopCmd, v => v.menuMoveTop).DisposeWith(disposables); | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.MoveUpCmd, v => v.menuMoveUp).DisposeWith(disposables); | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.MoveDownCmd, v => v.menuMoveDown).DisposeWith(disposables); | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.MoveBottomCmd, v => v.menuMoveBottom).DisposeWith(disposables); | ||||||
|  | 
 | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.SaveCmd, v => v.btnSave).DisposeWith(disposables); | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         // Context menu actions that require custom logic (Add, SelectAll) | ||||||
|  |         menuAddChildServer.Click += MenuAddChild_Click; | ||||||
|  |         menuSelectAllChild.Click += (s, e) => lstChild.SelectAll(); | ||||||
|  | 
 | ||||||
|  |         // Keyboard shortcuts when focus is within grid | ||||||
|  |         this.AddHandler(KeyDownEvent, AddGroupServerWindow_KeyDown, RoutingStrategies.Tunnel); | ||||||
|  |         lstChild.LoadingRow += LstChild_LoadingRow; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void LstChild_LoadingRow(object? sender, DataGridRowEventArgs e) | ||||||
|  |     { | ||||||
|  |         e.Row.Header = $" {e.Row.Index + 1}"; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private async Task<bool> UpdateViewHandler(EViewAction action, object? obj) | ||||||
|  |     { | ||||||
|  |         switch (action) | ||||||
|  |         { | ||||||
|  |             case EViewAction.CloseWindow: | ||||||
|  |                 this.Close(true); | ||||||
|  |                 break; | ||||||
|  |         } | ||||||
|  |         return await Task.FromResult(true); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void Window_Loaded(object? sender, RoutedEventArgs e) | ||||||
|  |     { | ||||||
|  |         txtRemarks.Focus(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void AddGroupServerWindow_KeyDown(object? sender, KeyEventArgs e) | ||||||
|  |     { | ||||||
|  |         if (!lstChild.IsKeyboardFocusWithin) | ||||||
|  |             return; | ||||||
|  | 
 | ||||||
|  |         if ((e.KeyModifiers & (KeyModifiers.Control | KeyModifiers.Meta)) != 0) | ||||||
|  |         { | ||||||
|  |             if (e.Key == Key.A) | ||||||
|  |             { | ||||||
|  |                 lstChild.SelectAll(); | ||||||
|  |                 e.Handled = true; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             switch (e.Key) | ||||||
|  |             { | ||||||
|  |                 case Key.T: | ||||||
|  |                     ViewModel?.MoveServer(EMove.Top); | ||||||
|  |                     e.Handled = true; | ||||||
|  |                     break; | ||||||
|  |                 case Key.U: | ||||||
|  |                     ViewModel?.MoveServer(EMove.Up); | ||||||
|  |                     e.Handled = true; | ||||||
|  |                     break; | ||||||
|  |                 case Key.D: | ||||||
|  |                     ViewModel?.MoveServer(EMove.Down); | ||||||
|  |                     e.Handled = true; | ||||||
|  |                     break; | ||||||
|  |                 case Key.B: | ||||||
|  |                     ViewModel?.MoveServer(EMove.Bottom); | ||||||
|  |                     e.Handled = true; | ||||||
|  |                     break; | ||||||
|  |                 case Key.Delete: | ||||||
|  |                     ViewModel?.ChildRemoveAsync(); | ||||||
|  |                     e.Handled = true; | ||||||
|  |                     break; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private async void MenuAddChild_Click(object? sender, RoutedEventArgs e) | ||||||
|  |     { | ||||||
|  |         var selectWindow = new ProfilesSelectWindow(); | ||||||
|  |         if (ViewModel?.SelectedSource?.ConfigType == EConfigType.PolicyGroup) | ||||||
|  |         { | ||||||
|  |             selectWindow.SetConfigTypeFilter(new[] { EConfigType.Custom }, exclude: true); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             selectWindow.SetConfigTypeFilter(new[] { EConfigType.Custom, EConfigType.PolicyGroup, EConfigType.ProxyChain }, exclude: true); | ||||||
|  |         } | ||||||
|  |         selectWindow.AllowMultiSelect(true); | ||||||
|  |         var result = await selectWindow.ShowDialog<bool?>(this); | ||||||
|  |         if (result == true) | ||||||
|  |         { | ||||||
|  |             var profiles = await selectWindow.ProfileItems; | ||||||
|  |             ViewModel?.ChildItemsObs.AddRange(profiles); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void LstChild_SelectionChanged(object? sender, SelectionChangedEventArgs e) | ||||||
|  |     { | ||||||
|  |         if (ViewModel != null) | ||||||
|  |         { | ||||||
|  |             ViewModel.SelectedChildren = lstChild.SelectedItems.Cast<ProfileItem>().ToList(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -35,6 +35,8 @@ | ||||||
|                         <MenuItem x:Name="menuAddServerViaScan" Header="{x:Static resx:ResUI.menuAddServerViaScan}" /> |                         <MenuItem x:Name="menuAddServerViaScan" Header="{x:Static resx:ResUI.menuAddServerViaScan}" /> | ||||||
|                         <MenuItem x:Name="menuAddServerViaImage" Header="{x:Static resx:ResUI.menuAddServerViaImage}" /> |                         <MenuItem x:Name="menuAddServerViaImage" Header="{x:Static resx:ResUI.menuAddServerViaImage}" /> | ||||||
|                         <MenuItem x:Name="menuAddCustomServer" Header="{x:Static resx:ResUI.menuAddCustomServer}" /> |                         <MenuItem x:Name="menuAddCustomServer" Header="{x:Static resx:ResUI.menuAddCustomServer}" /> | ||||||
|  |                         <MenuItem x:Name="menuAddPolicyGroupServer" Header="{x:Static resx:ResUI.menuAddPolicyGroupServer}" /> | ||||||
|  |                         <MenuItem x:Name="menuAddProxyChainServer" Header="{x:Static resx:ResUI.menuAddProxyChainServer}" /> | ||||||
|                         <Separator /> |                         <Separator /> | ||||||
|                         <MenuItem x:Name="menuAddVmessServer" Header="{x:Static resx:ResUI.menuAddVmessServer}" /> |                         <MenuItem x:Name="menuAddVmessServer" Header="{x:Static resx:ResUI.menuAddVmessServer}" /> | ||||||
|                         <MenuItem x:Name="menuAddVlessServer" Header="{x:Static resx:ResUI.menuAddVlessServer}" /> |                         <MenuItem x:Name="menuAddVlessServer" Header="{x:Static resx:ResUI.menuAddVlessServer}" /> | ||||||
|  |  | ||||||
|  | @ -83,6 +83,8 @@ public partial class MainWindow : WindowBase<MainWindowViewModel> | ||||||
|             this.BindCommand(ViewModel, vm => vm.AddWireguardServerCmd, v => v.menuAddWireguardServer).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.AddWireguardServerCmd, v => v.menuAddWireguardServer).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.AddAnytlsServerCmd, v => v.menuAddAnytlsServer).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.AddAnytlsServerCmd, v => v.menuAddAnytlsServer).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.AddCustomServerCmd, v => v.menuAddCustomServer).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.AddCustomServerCmd, v => v.menuAddCustomServer).DisposeWith(disposables); | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.AddPolicyGroupServerCmd, v => v.menuAddPolicyGroupServer).DisposeWith(disposables); | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.AddProxyChainServerCmd, v => v.menuAddProxyChainServer).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.AddServerViaClipboardCmd, v => v.menuAddServerViaClipboard).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.AddServerViaClipboardCmd, v => v.menuAddServerViaClipboard).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.AddServerViaScanCmd, v => v.menuAddServerViaScan).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.AddServerViaScanCmd, v => v.menuAddServerViaScan).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.AddServerViaImageCmd, v => v.menuAddServerViaImage).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.AddServerViaImageCmd, v => v.menuAddServerViaImage).DisposeWith(disposables); | ||||||
|  | @ -207,6 +209,11 @@ public partial class MainWindow : WindowBase<MainWindowViewModel> | ||||||
|                     return false; |                     return false; | ||||||
|                 return await new AddServer2Window((ProfileItem)obj).ShowDialog<bool>(this); |                 return await new AddServer2Window((ProfileItem)obj).ShowDialog<bool>(this); | ||||||
| 
 | 
 | ||||||
|  |             case EViewAction.AddGroupServerWindow: | ||||||
|  |                 if (obj is null) | ||||||
|  |                     return false; | ||||||
|  |                 return await new AddGroupServerWindow((ProfileItem)obj).ShowDialog<bool>(this); | ||||||
|  | 
 | ||||||
|             case EViewAction.DNSSettingWindow: |             case EViewAction.DNSSettingWindow: | ||||||
|                 return await new DNSSettingWindow().ShowDialog<bool>(this); |                 return await new DNSSettingWindow().ShowDialog<bool>(this); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -2,6 +2,7 @@ | ||||||
|     x:Class="v2rayN.Desktop.Views.MsgView" |     x:Class="v2rayN.Desktop.Views.MsgView" | ||||||
|     xmlns="https://github.com/avaloniaui" |     xmlns="https://github.com/avaloniaui" | ||||||
|     xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" |     xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" | ||||||
|  |     xmlns:avaloniaEdit="clr-namespace:AvaloniaEdit;assembly=AvaloniaEdit" | ||||||
|     xmlns:d="http://schemas.microsoft.com/expression/blend/2008" |     xmlns:d="http://schemas.microsoft.com/expression/blend/2008" | ||||||
|     xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" |     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" | ||||||
|  | @ -70,35 +71,36 @@ | ||||||
|                 Theme="{DynamicResource SimpleToggleSwitch}" /> |                 Theme="{DynamicResource SimpleToggleSwitch}" /> | ||||||
|         </WrapPanel> |         </WrapPanel> | ||||||
| 
 | 
 | ||||||
|         <ScrollViewer x:Name="msgScrollViewer" VerticalScrollBarVisibility="Auto"> |         <avaloniaEdit:TextEditor | ||||||
|             <SelectableTextBlock |             Name="txtMsg" | ||||||
|                 Name="txtMsg" |             Margin="{StaticResource Margin8}" | ||||||
|                 Margin="{StaticResource Margin8}" |             IsReadOnly="True" | ||||||
|                 VerticalAlignment="Stretch" |             VerticalScrollBarVisibility="Auto" | ||||||
|                 Classes="TextArea" |             WordWrap="True"> | ||||||
|                 TextAlignment="Left" |             <avaloniaEdit:TextEditor.Options> | ||||||
|                 TextWrapping="Wrap"> |                 <avaloniaEdit:TextEditorOptions AllowScrollBelowDocument="False"/> | ||||||
|                 <SelectableTextBlock.ContextMenu> |             </avaloniaEdit:TextEditor.Options> | ||||||
|                     <ContextMenu> |             <avaloniaEdit:TextEditor.ContextFlyout> | ||||||
|                         <MenuItem |                 <MenuFlyout> | ||||||
|                             x:Name="menuMsgViewSelectAll" |                     <MenuItem | ||||||
|                             Click="menuMsgViewSelectAll_Click" |                         x:Name="menuMsgViewSelectAll" | ||||||
|                             Header="{x:Static resx:ResUI.menuMsgViewSelectAll}" /> |                         Click="menuMsgViewSelectAll_Click" | ||||||
|                         <MenuItem |                         Header="{x:Static resx:ResUI.menuMsgViewSelectAll}" /> | ||||||
|                             x:Name="menuMsgViewCopy" |                     <MenuItem | ||||||
|                             Click="menuMsgViewCopy_Click" |                         x:Name="menuMsgViewCopy" | ||||||
|                             Header="{x:Static resx:ResUI.menuMsgViewCopy}" /> |                         Click="menuMsgViewCopy_Click" | ||||||
|                         <MenuItem |                         Header="{x:Static resx:ResUI.menuMsgViewCopy}" /> | ||||||
|                             x:Name="menuMsgViewCopyAll" |                     <MenuItem | ||||||
|                             Click="menuMsgViewCopyAll_Click" |                         x:Name="menuMsgViewCopyAll" | ||||||
|                             Header="{x:Static resx:ResUI.menuMsgViewCopyAll}" /> |                         Click="menuMsgViewCopyAll_Click" | ||||||
|                         <MenuItem |                         Header="{x:Static resx:ResUI.menuMsgViewCopyAll}" /> | ||||||
|                             x:Name="menuMsgViewClear" |                     <MenuItem | ||||||
|                             Click="menuMsgViewClear_Click" |                         x:Name="menuMsgViewClear" | ||||||
|                             Header="{x:Static resx:ResUI.menuMsgViewClear}" /> |                         Click="menuMsgViewClear_Click" | ||||||
|                     </ContextMenu> |                         Header="{x:Static resx:ResUI.menuMsgViewClear}" /> | ||||||
|                 </SelectableTextBlock.ContextMenu> |                 </MenuFlyout> | ||||||
|             </SelectableTextBlock> |             </avaloniaEdit:TextEditor.ContextFlyout> | ||||||
|         </ScrollViewer> |         </avaloniaEdit:TextEditor> | ||||||
|  | 
 | ||||||
|     </DockPanel> |     </DockPanel> | ||||||
| </UserControl> | </UserControl> | ||||||
|  |  | ||||||
|  | @ -1,6 +1,6 @@ | ||||||
| using System.Reactive.Disposables; | using System.Reactive.Disposables; | ||||||
| using Avalonia.Controls; |  | ||||||
| using Avalonia.Interactivity; | using Avalonia.Interactivity; | ||||||
|  | using Avalonia.Media; | ||||||
| using Avalonia.ReactiveUI; | using Avalonia.ReactiveUI; | ||||||
| using Avalonia.Threading; | using Avalonia.Threading; | ||||||
| using ReactiveUI; | using ReactiveUI; | ||||||
|  | @ -10,13 +10,11 @@ namespace v2rayN.Desktop.Views; | ||||||
| 
 | 
 | ||||||
| public partial class MsgView : ReactiveUserControl<MsgViewModel> | public partial class MsgView : ReactiveUserControl<MsgViewModel> | ||||||
| { | { | ||||||
|     private readonly ScrollViewer _scrollViewer; |     //private const int KeepLines = 30; | ||||||
| 
 | 
 | ||||||
|     public MsgView() |     public MsgView() | ||||||
|     { |     { | ||||||
|         InitializeComponent(); |         InitializeComponent(); | ||||||
|         _scrollViewer = this.FindControl<ScrollViewer>("msgScrollViewer"); |  | ||||||
| 
 |  | ||||||
|         ViewModel = new MsgViewModel(UpdateViewHandler); |         ViewModel = new MsgViewModel(UpdateViewHandler); | ||||||
| 
 | 
 | ||||||
|         this.WhenActivated(disposables => |         this.WhenActivated(disposables => | ||||||
|  | @ -24,6 +22,11 @@ public partial class MsgView : ReactiveUserControl<MsgViewModel> | ||||||
|             this.Bind(ViewModel, vm => vm.MsgFilter, v => v.cmbMsgFilter.Text).DisposeWith(disposables); |             this.Bind(ViewModel, vm => vm.MsgFilter, v => v.cmbMsgFilter.Text).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); | ||||||
|         }); |         }); | ||||||
|  | 
 | ||||||
|  |         TextEditorKeywordHighlighter.Attach(txtMsg, Global.LogLevelColors.ToDictionary( | ||||||
|  |                 kv => kv.Key, | ||||||
|  |                 kv => (IBrush)new SolidColorBrush(Color.Parse(kv.Value)) | ||||||
|  |             )); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private async Task<bool> UpdateViewHandler(EViewAction action, object? obj) |     private async Task<bool> UpdateViewHandler(EViewAction action, object? obj) | ||||||
|  | @ -34,9 +37,8 @@ public partial class MsgView : ReactiveUserControl<MsgViewModel> | ||||||
|                 if (obj is null) |                 if (obj is null) | ||||||
|                     return false; |                     return false; | ||||||
| 
 | 
 | ||||||
|                 Dispatcher.UIThread.Post(() => |                 Dispatcher.UIThread.Post(() => ShowMsg(obj), | ||||||
|                     ShowMsg(obj), |                     DispatcherPriority.ApplicationIdle); | ||||||
|                    DispatcherPriority.ApplicationIdle); |  | ||||||
|                 break; |                 break; | ||||||
|         } |         } | ||||||
|         return await Task.FromResult(true); |         return await Task.FromResult(true); | ||||||
|  | @ -44,23 +46,41 @@ public partial class MsgView : ReactiveUserControl<MsgViewModel> | ||||||
| 
 | 
 | ||||||
|     private void ShowMsg(object msg) |     private void ShowMsg(object msg) | ||||||
|     { |     { | ||||||
|         txtMsg.Text = msg.ToString(); |         txtMsg.BeginChange(); | ||||||
|  | 
 | ||||||
|  |         //var lineCount = txtMsg.LineCount; | ||||||
|  |         //if (lineCount > ViewModel?.NumMaxMsg) | ||||||
|  |         //{ | ||||||
|  |         //    var cutLine = txtMsg.Document.GetLineByNumber(lineCount - KeepLines); | ||||||
|  |         //    txtMsg.Document.Remove(0, cutLine.Offset); | ||||||
|  |         //} | ||||||
|  |         if (txtMsg.LineCount > ViewModel?.NumMaxMsg) | ||||||
|  |         { | ||||||
|  |             ClearMsg(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         txtMsg.AppendText(msg.ToString()); | ||||||
|         if (togScrollToEnd.IsChecked ?? true) |         if (togScrollToEnd.IsChecked ?? true) | ||||||
|         { |         { | ||||||
|             _scrollViewer?.ScrollToEnd(); |             txtMsg.ScrollToEnd(); | ||||||
|         } |         } | ||||||
|  | 
 | ||||||
|  |         txtMsg.EndChange(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public void ClearMsg() |     public void ClearMsg() | ||||||
|     { |     { | ||||||
|         ViewModel?.ClearMsg(); |         txtMsg.Text = string.Empty; | ||||||
|         txtMsg.Text = ""; |         txtMsg.AppendText("----- Message cleared -----\n"); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private void menuMsgViewSelectAll_Click(object? sender, RoutedEventArgs e) |     private void menuMsgViewSelectAll_Click(object? sender, RoutedEventArgs e) | ||||||
|     { |     { | ||||||
|         txtMsg.Focus(); |         Dispatcher.UIThread.Post(() => | ||||||
|         txtMsg.SelectAll(); |         { | ||||||
|  |             txtMsg.TextArea.Focus(); | ||||||
|  |             txtMsg.SelectAll(); | ||||||
|  |         }, DispatcherPriority.Render); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private async void menuMsgViewCopy_Click(object? sender, RoutedEventArgs e) |     private async void menuMsgViewCopy_Click(object? sender, RoutedEventArgs e) | ||||||
|  |  | ||||||
|  | @ -99,13 +99,14 @@ | ||||||
|                         <MenuItem x:Name="menuCopyServer" Header="{x:Static resx:ResUI.menuCopyServer}" /> |                         <MenuItem x:Name="menuCopyServer" Header="{x:Static resx:ResUI.menuCopyServer}" /> | ||||||
|                         <MenuItem x:Name="menuShareServer" Header="{x:Static resx:ResUI.menuShareServer}" /> |                         <MenuItem x:Name="menuShareServer" Header="{x:Static resx:ResUI.menuShareServer}" /> | ||||||
|                         <Separator /> |                         <Separator /> | ||||||
|                         <MenuItem Header="{x:Static resx:ResUI.menuSetDefaultMultipleServer}"> |                         <MenuItem Header="{x:Static resx:ResUI.menuGenGroupMultipleServer}"> | ||||||
|                             <MenuItem x:Name="menuSetDefaultMultipleServerXrayRandom" Header="{x:Static resx:ResUI.menuSetDefaultMultipleServerXrayRandom}" /> |                             <MenuItem x:Name="menuGenGroupMultipleServerXrayRandom" Header="{x:Static resx:ResUI.menuGenGroupMultipleServerXrayRandom}" /> | ||||||
|                             <MenuItem x:Name="menuSetDefaultMultipleServerXrayRoundRobin" Header="{x:Static resx:ResUI.menuSetDefaultMultipleServerXrayRoundRobin}" /> |                             <MenuItem x:Name="menuGenGroupMultipleServerXrayRoundRobin" Header="{x:Static resx:ResUI.menuGenGroupMultipleServerXrayRoundRobin}" /> | ||||||
|                             <MenuItem x:Name="menuSetDefaultMultipleServerXrayLeastPing" Header="{x:Static resx:ResUI.menuSetDefaultMultipleServerXrayLeastPing}" /> |                             <MenuItem x:Name="menuGenGroupMultipleServerXrayLeastPing" Header="{x:Static resx:ResUI.menuGenGroupMultipleServerXrayLeastPing}" /> | ||||||
|                             <MenuItem x:Name="menuSetDefaultMultipleServerXrayLeastLoad" Header="{x:Static resx:ResUI.menuSetDefaultMultipleServerXrayLeastLoad}" /> |                             <MenuItem x:Name="menuGenGroupMultipleServerXrayLeastLoad" Header="{x:Static resx:ResUI.menuGenGroupMultipleServerXrayLeastLoad}" /> | ||||||
|                             <Separator /> |                             <Separator /> | ||||||
|                             <MenuItem x:Name="menuSetDefaultMultipleServerSingBoxLeastPing" Header="{x:Static resx:ResUI.menuSetDefaultMultipleServerSingBoxLeastPing}" /> |                             <MenuItem x:Name="menuGenGroupMultipleServerSingBoxLeastPing" Header="{x:Static resx:ResUI.menuGenGroupMultipleServerSingBoxLeastPing}" /> | ||||||
|  |                             <MenuItem x:Name="menuGenGroupMultipleServerSingBoxFallback" Header="{x:Static resx:ResUI.menuGenGroupMultipleServerSingBoxFallback}" /> | ||||||
|                         </MenuItem> |                         </MenuItem> | ||||||
|                         <Separator /> |                         <Separator /> | ||||||
|                         <MenuItem x:Name="menuMixedTestServer" Header="{x:Static resx:ResUI.menuMixedTestServer}" /> |                         <MenuItem x:Name="menuMixedTestServer" Header="{x:Static resx:ResUI.menuMixedTestServer}" /> | ||||||
|  |  | ||||||
|  | @ -66,11 +66,12 @@ public partial class ProfilesView : ReactiveUserControl<ProfilesViewModel> | ||||||
|             this.BindCommand(ViewModel, vm => vm.CopyServerCmd, v => v.menuCopyServer).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.CopyServerCmd, v => v.menuCopyServer).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.SetDefaultServerCmd, v => v.menuSetDefaultServer).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.SetDefaultServerCmd, v => v.menuSetDefaultServer).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.ShareServerCmd, v => v.menuShareServer).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.ShareServerCmd, v => v.menuShareServer).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.SetDefaultMultipleServerXrayRandomCmd, v => v.menuSetDefaultMultipleServerXrayRandom).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.GenGroupMultipleServerXrayRandomCmd, v => v.menuGenGroupMultipleServerXrayRandom).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.SetDefaultMultipleServerXrayRoundRobinCmd, v => v.menuSetDefaultMultipleServerXrayRoundRobin).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.GenGroupMultipleServerXrayRoundRobinCmd, v => v.menuGenGroupMultipleServerXrayRoundRobin).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.SetDefaultMultipleServerXrayLeastPingCmd, v => v.menuSetDefaultMultipleServerXrayLeastPing).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.GenGroupMultipleServerXrayLeastPingCmd, v => v.menuGenGroupMultipleServerXrayLeastPing).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.SetDefaultMultipleServerXrayLeastLoadCmd, v => v.menuSetDefaultMultipleServerXrayLeastLoad).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.GenGroupMultipleServerXrayLeastLoadCmd, v => v.menuGenGroupMultipleServerXrayLeastLoad).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.SetDefaultMultipleServerSingBoxLeastPingCmd, v => v.menuSetDefaultMultipleServerSingBoxLeastPing).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.GenGroupMultipleServerSingBoxLeastPingCmd, v => v.menuGenGroupMultipleServerSingBoxLeastPing).DisposeWith(disposables); | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.GenGroupMultipleServerSingBoxFallbackCmd, v => v.menuGenGroupMultipleServerSingBoxFallback).DisposeWith(disposables); | ||||||
| 
 | 
 | ||||||
|             //servers move |             //servers move | ||||||
|             //this.OneWayBind(ViewModel, vm => vm.SubItems, v => v.cmbMoveToGroup.ItemsSource).DisposeWith(disposables); |             //this.OneWayBind(ViewModel, vm => vm.SubItems, v => v.cmbMoveToGroup.ItemsSource).DisposeWith(disposables); | ||||||
|  | @ -167,6 +168,11 @@ public partial class ProfilesView : ReactiveUserControl<ProfilesViewModel> | ||||||
|                     return false; |                     return false; | ||||||
|                 return await new AddServer2Window((ProfileItem)obj).ShowDialog<bool>(_window); |                 return await new AddServer2Window((ProfileItem)obj).ShowDialog<bool>(_window); | ||||||
| 
 | 
 | ||||||
|  |             case EViewAction.AddGroupServerWindow: | ||||||
|  |                 if (obj is null) | ||||||
|  |                     return false; | ||||||
|  |                 return await new AddGroupServerWindow((ProfileItem)obj).ShowDialog<bool>(_window); | ||||||
|  | 
 | ||||||
|             case EViewAction.ShareServer: |             case EViewAction.ShareServer: | ||||||
|                 if (obj is null) |                 if (obj is null) | ||||||
|                     return false; |                     return false; | ||||||
|  |  | ||||||
|  | @ -63,7 +63,7 @@ public partial class SubEditWindow : WindowBase<SubEditViewModel> | ||||||
|     private async void BtnSelectPrevProfile_Click(object? sender, RoutedEventArgs e) |     private async void BtnSelectPrevProfile_Click(object? sender, RoutedEventArgs e) | ||||||
|     { |     { | ||||||
|         var selectWindow = new ProfilesSelectWindow(); |         var selectWindow = new ProfilesSelectWindow(); | ||||||
|         selectWindow.SetConfigTypeFilter(new[] { EConfigType.Custom }, exclude: true); |         selectWindow.SetConfigTypeFilter(new[] { EConfigType.Custom, EConfigType.PolicyGroup, EConfigType.ProxyChain }, exclude: true); | ||||||
|         var result = await selectWindow.ShowDialog<bool?>(this); |         var result = await selectWindow.ShowDialog<bool?>(this); | ||||||
|         if (result == true) |         if (result == true) | ||||||
|         { |         { | ||||||
|  | @ -78,7 +78,7 @@ public partial class SubEditWindow : WindowBase<SubEditViewModel> | ||||||
|     private async void BtnSelectNextProfile_Click(object? sender, RoutedEventArgs e) |     private async void BtnSelectNextProfile_Click(object? sender, RoutedEventArgs e) | ||||||
|     { |     { | ||||||
|         var selectWindow = new ProfilesSelectWindow(); |         var selectWindow = new ProfilesSelectWindow(); | ||||||
|         selectWindow.SetConfigTypeFilter(new[] { EConfigType.Custom }, exclude: true); |         selectWindow.SetConfigTypeFilter(new[] { EConfigType.Custom, EConfigType.PolicyGroup, EConfigType.ProxyChain }, exclude: true); | ||||||
|         var result = await selectWindow.ShowDialog<bool?>(this); |         var result = await selectWindow.ShowDialog<bool?>(this); | ||||||
|         if (result == true) |         if (result == true) | ||||||
|         { |         { | ||||||
|  |  | ||||||
|  | @ -9,6 +9,7 @@ | ||||||
| 	</PropertyGroup> | 	</PropertyGroup> | ||||||
| 
 | 
 | ||||||
| 	<ItemGroup> | 	<ItemGroup> | ||||||
|  | 		<PackageReference Include="Avalonia.AvaloniaEdit" /> | ||||||
| 		<PackageReference Include="Avalonia.Controls.DataGrid"> | 		<PackageReference Include="Avalonia.Controls.DataGrid"> | ||||||
| 			<TreatAsUsed>true</TreatAsUsed> | 			<TreatAsUsed>true</TreatAsUsed> | ||||||
| 		</PackageReference> | 		</PackageReference> | ||||||
|  | @ -17,6 +18,7 @@ | ||||||
| 		<PackageReference Include="Avalonia.ReactiveUI" /> | 		<PackageReference Include="Avalonia.ReactiveUI" /> | ||||||
| 		<PackageReference Include="MessageBox.Avalonia" /> | 		<PackageReference Include="MessageBox.Avalonia" /> | ||||||
| 		<PackageReference Include="Semi.Avalonia" /> | 		<PackageReference Include="Semi.Avalonia" /> | ||||||
|  | 		<PackageReference Include="Semi.Avalonia.AvaloniaEdit" /> | ||||||
| 		<PackageReference Include="Semi.Avalonia.DataGrid"> | 		<PackageReference Include="Semi.Avalonia.DataGrid"> | ||||||
| 			<TreatAsUsed>true</TreatAsUsed> | 			<TreatAsUsed>true</TreatAsUsed> | ||||||
| 		</PackageReference> | 		</PackageReference> | ||||||
|  |  | ||||||
							
								
								
									
										213
									
								
								v2rayN/v2rayN/Views/AddGroupServerWindow.xaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										213
									
								
								v2rayN/v2rayN/Views/AddGroupServerWindow.xaml
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,213 @@ | ||||||
|  | <base:WindowBase | ||||||
|  |     x:Class="v2rayN.Views.AddGroupServerWindow" | ||||||
|  |     xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" | ||||||
|  |     xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" | ||||||
|  |     xmlns:base="clr-namespace:v2rayN.Base" | ||||||
|  |     xmlns:d="http://schemas.microsoft.com/expression/blend/2008" | ||||||
|  |     xmlns:materialDesign="http://materialdesigninxaml.net/winfx/xaml/themes" | ||||||
|  |     xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" | ||||||
|  |     xmlns:reactiveui="http://reactiveui.net" | ||||||
|  |     xmlns:resx="clr-namespace:ServiceLib.Resx;assembly=ServiceLib" | ||||||
|  |     xmlns:vms="clr-namespace:ServiceLib.ViewModels;assembly=ServiceLib" | ||||||
|  |     Title="{x:Static resx:ResUI.menuServers}" | ||||||
|  |     Width="900" | ||||||
|  |     Height="700" | ||||||
|  |     x:TypeArguments="vms:AddGroupServerViewModel" | ||||||
|  |     ResizeMode="CanResize" | ||||||
|  |     ShowInTaskbar="False" | ||||||
|  |     Style="{StaticResource WindowGlobal}" | ||||||
|  |     WindowStartupLocation="CenterScreen" | ||||||
|  |     mc:Ignorable="d"> | ||||||
|  | 
 | ||||||
|  |     <DockPanel Margin="{StaticResource Margin8}"> | ||||||
|  |         <StackPanel | ||||||
|  |             Margin="{StaticResource Margin4}" | ||||||
|  |             HorizontalAlignment="Center" | ||||||
|  |             DockPanel.Dock="Bottom" | ||||||
|  |             Orientation="Horizontal"> | ||||||
|  |             <Button | ||||||
|  |                 x:Name="btnSave" | ||||||
|  |                 Width="100" | ||||||
|  |                 Content="{x:Static resx:ResUI.TbConfirm}" | ||||||
|  |                 IsDefault="True" | ||||||
|  |                 Style="{StaticResource DefButton}" /> | ||||||
|  |             <Button | ||||||
|  |                 x:Name="btnCancel" | ||||||
|  |                 Width="100" | ||||||
|  |                 Margin="{StaticResource MarginLeftRight8}" | ||||||
|  |                 Content="{x:Static resx:ResUI.TbCancel}" | ||||||
|  |                 IsCancel="true" | ||||||
|  |                 Style="{StaticResource DefButton}" /> | ||||||
|  |         </StackPanel> | ||||||
|  |         <Grid DockPanel.Dock="Top"> | ||||||
|  |             <Grid.RowDefinitions> | ||||||
|  |                 <RowDefinition Height="Auto" /> | ||||||
|  |                 <RowDefinition Height="*" /> | ||||||
|  |                 <RowDefinition Height="Auto" /> | ||||||
|  |                 <RowDefinition Height="Auto" /> | ||||||
|  |                 <RowDefinition Height="Auto" /> | ||||||
|  |                 <RowDefinition Height="Auto" /> | ||||||
|  |                 <RowDefinition Height="Auto" /> | ||||||
|  |                 <RowDefinition Height="Auto" /> | ||||||
|  |                 <RowDefinition Height="Auto" /> | ||||||
|  |             </Grid.RowDefinitions> | ||||||
|  | 
 | ||||||
|  |             <Grid Grid.Row="0"> | ||||||
|  |                 <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" /> | ||||||
|  |                     <ColumnDefinition Width="Auto" /> | ||||||
|  |                 </Grid.ColumnDefinitions> | ||||||
|  | 
 | ||||||
|  |                 <TextBlock | ||||||
|  |                     Grid.Row="0" | ||||||
|  |                     Grid.Column="0" | ||||||
|  |                     Margin="{StaticResource Margin4}" | ||||||
|  |                     Style="{StaticResource ModuleTitle}" | ||||||
|  |                     Text="{x:Static resx:ResUI.menuServers}" /> | ||||||
|  | 
 | ||||||
|  |                 <TextBlock | ||||||
|  |                     Grid.Row="1" | ||||||
|  |                     Grid.Column="0" | ||||||
|  |                     Margin="{StaticResource Margin4}" | ||||||
|  |                     VerticalAlignment="Center" | ||||||
|  |                     Style="{StaticResource ToolbarTextBlock}" | ||||||
|  |                     Text="{x:Static resx:ResUI.TbRemarks}" /> | ||||||
|  |                 <TextBox | ||||||
|  |                     x:Name="txtRemarks" | ||||||
|  |                     Grid.Row="1" | ||||||
|  |                     Grid.Column="1" | ||||||
|  |                     Width="400" | ||||||
|  |                     Margin="{StaticResource Margin4}" | ||||||
|  |                     Style="{StaticResource DefTextBox}" /> | ||||||
|  | 
 | ||||||
|  |                 <TextBlock | ||||||
|  |                     Grid.Row="2" | ||||||
|  |                     Grid.Column="0" | ||||||
|  |                     Margin="{StaticResource Margin4}" | ||||||
|  |                     VerticalAlignment="Center" | ||||||
|  |                     Style="{StaticResource ToolbarTextBlock}" | ||||||
|  |                     Text="{x:Static resx:ResUI.TbCoreType}" /> | ||||||
|  |                 <ComboBox | ||||||
|  |                     x:Name="cmbCoreType" | ||||||
|  |                     Grid.Row="2" | ||||||
|  |                     Grid.Column="1" | ||||||
|  |                     Width="200" | ||||||
|  |                     Margin="{StaticResource Margin4}" | ||||||
|  |                     materialDesign:HintAssist.Hint="{x:Static resx:ResUI.TbCoreType}" | ||||||
|  |                     Style="{StaticResource DefComboBox}" /> | ||||||
|  | 
 | ||||||
|  |                 <Grid | ||||||
|  |                     x:Name="gridPolicyGroup" | ||||||
|  |                     Grid.Row="3" | ||||||
|  |                     Grid.Column="0" | ||||||
|  |                     Grid.ColumnSpan="3"> | ||||||
|  |                     <Grid.ColumnDefinitions> | ||||||
|  |                         <ColumnDefinition Width="180" /> | ||||||
|  |                         <ColumnDefinition Width="Auto" /> | ||||||
|  |                         <ColumnDefinition Width="Auto" /> | ||||||
|  |                     </Grid.ColumnDefinitions> | ||||||
|  |                     <TextBlock | ||||||
|  |                         Grid.Row="3" | ||||||
|  |                         Grid.Column="0" | ||||||
|  |                         Margin="{StaticResource Margin4}" | ||||||
|  |                         VerticalAlignment="Center" | ||||||
|  |                         Style="{StaticResource ToolbarTextBlock}" | ||||||
|  |                         Text="{x:Static resx:ResUI.TbPolicyGroupType}" /> | ||||||
|  |                     <ComboBox | ||||||
|  |                         x:Name="cmbPolicyGroupType" | ||||||
|  |                         Grid.Row="3" | ||||||
|  |                         Grid.Column="1" | ||||||
|  |                         Width="200" | ||||||
|  |                         Margin="{StaticResource Margin4}" | ||||||
|  |                         materialDesign:HintAssist.Hint="{x:Static resx:ResUI.TbPolicyGroupType}" | ||||||
|  |                         Style="{StaticResource DefComboBox}" /> | ||||||
|  |                 </Grid> | ||||||
|  |             </Grid> | ||||||
|  |         </Grid> | ||||||
|  |         <TabControl> | ||||||
|  |             <TabItem HorizontalAlignment="Left" Header="{x:Static resx:ResUI.menuServerList}"> | ||||||
|  |                 <DataGrid | ||||||
|  |                     x:Name="lstChild" | ||||||
|  |                     AutoGenerateColumns="False" | ||||||
|  |                     BorderThickness="1" | ||||||
|  |                     CanUserAddRows="False" | ||||||
|  |                     CanUserResizeRows="False" | ||||||
|  |                     CanUserSortColumns="False" | ||||||
|  |                     EnableRowVirtualization="True" | ||||||
|  |                     GridLinesVisibility="All" | ||||||
|  |                     HeadersVisibility="Column" | ||||||
|  |                     IsReadOnly="True" | ||||||
|  |                     Style="{StaticResource DefDataGrid}"> | ||||||
|  |                     <DataGrid.ContextMenu> | ||||||
|  |                         <ContextMenu Style="{StaticResource DefContextMenu}"> | ||||||
|  |                             <MenuItem | ||||||
|  |                                 x:Name="menuAddChildServer" | ||||||
|  |                                 Height="{StaticResource MenuItemHeight}" | ||||||
|  |                                 Click="MenuAddChild_Click" | ||||||
|  |                                 Header="{x:Static resx:ResUI.menuAddChildServer}" /> | ||||||
|  |                             <MenuItem | ||||||
|  |                                 x:Name="menuRemoveChildServer" | ||||||
|  |                                 Height="{StaticResource MenuItemHeight}" | ||||||
|  |                                 Header="{x:Static resx:ResUI.menuRemoveChildServer}" /> | ||||||
|  |                             <MenuItem | ||||||
|  |                                 x:Name="menuSelectAllChild" | ||||||
|  |                                 Height="{StaticResource MenuItemHeight}" | ||||||
|  |                                 Header="{x:Static resx:ResUI.menuSelectAll}" /> | ||||||
|  |                             <Separator /> | ||||||
|  |                             <MenuItem | ||||||
|  |                                 x:Name="menuMoveTop" | ||||||
|  |                                 Height="{StaticResource MenuItemHeight}" | ||||||
|  |                                 Header="{x:Static resx:ResUI.menuMoveTop}" /> | ||||||
|  |                             <MenuItem | ||||||
|  |                                 x:Name="menuMoveUp" | ||||||
|  |                                 Height="{StaticResource MenuItemHeight}" | ||||||
|  |                                 Header="{x:Static resx:ResUI.menuMoveUp}" /> | ||||||
|  |                             <MenuItem | ||||||
|  |                                 x:Name="menuMoveDown" | ||||||
|  |                                 Height="{StaticResource MenuItemHeight}" | ||||||
|  |                                 Header="{x:Static resx:ResUI.menuMoveDown}" /> | ||||||
|  |                             <MenuItem | ||||||
|  |                                 x:Name="menuMoveBottom" | ||||||
|  |                                 Height="{StaticResource MenuItemHeight}" | ||||||
|  |                                 Header="{x:Static resx:ResUI.menuMoveBottom}" /> | ||||||
|  |                         </ContextMenu> | ||||||
|  |                     </DataGrid.ContextMenu> | ||||||
|  |                     <DataGrid.Columns> | ||||||
|  |                         <DataGridTextColumn | ||||||
|  |                             Width="150" | ||||||
|  |                             Binding="{Binding ConfigType}" | ||||||
|  |                             Header="{x:Static resx:ResUI.LvServiceType}" /> | ||||||
|  |                         <DataGridTextColumn | ||||||
|  |                             Width="150" | ||||||
|  |                             Binding="{Binding Remarks}" | ||||||
|  |                             Header="{x:Static resx:ResUI.LvRemarks}" /> | ||||||
|  |                         <DataGridTextColumn | ||||||
|  |                             Width="120" | ||||||
|  |                             Binding="{Binding Address}" | ||||||
|  |                             Header="{x:Static resx:ResUI.LvAddress}" /> | ||||||
|  |                         <DataGridTextColumn | ||||||
|  |                             Width="100" | ||||||
|  |                             Binding="{Binding Port}" | ||||||
|  |                             Header="{x:Static resx:ResUI.LvPort}" /> | ||||||
|  |                         <DataGridTextColumn | ||||||
|  |                             Width="100" | ||||||
|  |                             Binding="{Binding Network}" | ||||||
|  |                             Header="{x:Static resx:ResUI.LvTransportProtocol}" /> | ||||||
|  |                         <DataGridTextColumn | ||||||
|  |                             Width="100" | ||||||
|  |                             Binding="{Binding StreamSecurity}" | ||||||
|  |                             Header="{x:Static resx:ResUI.LvTLS}" /> | ||||||
|  |                     </DataGrid.Columns> | ||||||
|  |                 </DataGrid> | ||||||
|  |             </TabItem> | ||||||
|  |         </TabControl> | ||||||
|  |     </DockPanel> | ||||||
|  | </base:WindowBase> | ||||||
							
								
								
									
										148
									
								
								v2rayN/v2rayN/Views/AddGroupServerWindow.xaml.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										148
									
								
								v2rayN/v2rayN/Views/AddGroupServerWindow.xaml.cs
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,148 @@ | ||||||
|  | using System.Reactive.Disposables; | ||||||
|  | using System.Windows; | ||||||
|  | using System.Windows.Input; | ||||||
|  | using DynamicData; | ||||||
|  | using ReactiveUI; | ||||||
|  | 
 | ||||||
|  | namespace v2rayN.Views; | ||||||
|  | 
 | ||||||
|  | public partial class AddGroupServerWindow | ||||||
|  | { | ||||||
|  |     public AddGroupServerWindow(ProfileItem profileItem) | ||||||
|  |     { | ||||||
|  |         InitializeComponent(); | ||||||
|  | 
 | ||||||
|  |         this.Owner = Application.Current.MainWindow; | ||||||
|  |         this.Loaded += Window_Loaded; | ||||||
|  |         this.PreviewKeyDown += AddGroupServerWindow_PreviewKeyDown; | ||||||
|  |         lstChild.SelectionChanged += LstChild_SelectionChanged; | ||||||
|  |         menuSelectAllChild.Click += MenuSelectAllChild_Click; | ||||||
|  | 
 | ||||||
|  |         ViewModel = new AddGroupServerViewModel(profileItem, UpdateViewHandler); | ||||||
|  | 
 | ||||||
|  |         cmbCoreType.ItemsSource = Global.CoreTypes; | ||||||
|  |         cmbPolicyGroupType.ItemsSource = new List<string> | ||||||
|  |         { | ||||||
|  |             ResUI.TbLeastPing, | ||||||
|  |             ResUI.TbFallback, | ||||||
|  |             ResUI.TbRandom, | ||||||
|  |             ResUI.TbRoundRobin, | ||||||
|  |             ResUI.TbLeastLoad, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         switch (profileItem.ConfigType) | ||||||
|  |         { | ||||||
|  |             case EConfigType.PolicyGroup: | ||||||
|  |                 this.Title = ResUI.TbConfigTypePolicyGroup; | ||||||
|  |                 break; | ||||||
|  | 
 | ||||||
|  |             case EConfigType.ProxyChain: | ||||||
|  |                 this.Title = ResUI.TbConfigTypeProxyChain; | ||||||
|  |                 gridPolicyGroup.Visibility = Visibility.Collapsed; | ||||||
|  |                 break; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         this.WhenActivated(disposables => | ||||||
|  |         { | ||||||
|  |             this.Bind(ViewModel, vm => vm.SelectedSource.Remarks, v => v.txtRemarks.Text).DisposeWith(disposables); | ||||||
|  |             this.Bind(ViewModel, vm => vm.CoreType, v => v.cmbCoreType.Text).DisposeWith(disposables); | ||||||
|  |             this.Bind(ViewModel, vm => vm.PolicyGroupType, v => v.cmbPolicyGroupType.Text).DisposeWith(disposables); | ||||||
|  | 
 | ||||||
|  |             this.OneWayBind(ViewModel, vm => vm.ChildItemsObs, v => v.lstChild.ItemsSource).DisposeWith(disposables); | ||||||
|  |             this.Bind(ViewModel, vm => vm.SelectedChild, v => v.lstChild.SelectedItem).DisposeWith(disposables); | ||||||
|  | 
 | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.RemoveCmd, v => v.menuRemoveChildServer).DisposeWith(disposables); | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.MoveTopCmd, v => v.menuMoveTop).DisposeWith(disposables); | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.MoveUpCmd, v => v.menuMoveUp).DisposeWith(disposables); | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.MoveDownCmd, v => v.menuMoveDown).DisposeWith(disposables); | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.MoveBottomCmd, v => v.menuMoveBottom).DisposeWith(disposables); | ||||||
|  | 
 | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.SaveCmd, v => v.btnSave).DisposeWith(disposables); | ||||||
|  |         }); | ||||||
|  |         WindowsUtils.SetDarkBorder(this, AppManager.Instance.Config.UiItem.CurrentTheme); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private async Task<bool> UpdateViewHandler(EViewAction action, object? obj) | ||||||
|  |     { | ||||||
|  |         switch (action) | ||||||
|  |         { | ||||||
|  |             case EViewAction.CloseWindow: | ||||||
|  |                 this.DialogResult = true; | ||||||
|  |                 break; | ||||||
|  |         } | ||||||
|  |         return await Task.FromResult(true); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void Window_Loaded(object sender, RoutedEventArgs e) | ||||||
|  |     { | ||||||
|  |         txtRemarks.Focus(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void AddGroupServerWindow_PreviewKeyDown(object sender, KeyEventArgs e) | ||||||
|  |     { | ||||||
|  |         if (!lstChild.IsKeyboardFocusWithin) | ||||||
|  |             return; | ||||||
|  |         if (Keyboard.IsKeyDown(Key.LeftCtrl) || Keyboard.IsKeyDown(Key.RightCtrl)) | ||||||
|  |         { | ||||||
|  |             if (e.Key == Key.A) | ||||||
|  |             { | ||||||
|  |                 lstChild.SelectAll(); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             if (e.Key == Key.T) | ||||||
|  |             { | ||||||
|  |                 ViewModel?.MoveServer(EMove.Top); | ||||||
|  |             } | ||||||
|  |             else if (e.Key == Key.U) | ||||||
|  |             { | ||||||
|  |                 ViewModel?.MoveServer(EMove.Up); | ||||||
|  |             } | ||||||
|  |             else if (e.Key == Key.D) | ||||||
|  |             { | ||||||
|  |                 ViewModel?.MoveServer(EMove.Down); | ||||||
|  |             } | ||||||
|  |             else if (e.Key == Key.B) | ||||||
|  |             { | ||||||
|  |                 ViewModel?.MoveServer(EMove.Bottom); | ||||||
|  |             } | ||||||
|  |             else if (e.Key == Key.Delete) | ||||||
|  |             { | ||||||
|  |                 ViewModel?.ChildRemoveAsync(); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private async void MenuAddChild_Click(object sender, RoutedEventArgs e) | ||||||
|  |     { | ||||||
|  |         var selectWindow = new ProfilesSelectWindow(); | ||||||
|  |         if (ViewModel?.SelectedSource?.ConfigType == EConfigType.PolicyGroup) | ||||||
|  |         { | ||||||
|  |             selectWindow.SetConfigTypeFilter(new[] { EConfigType.Custom }, exclude: true); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             selectWindow.SetConfigTypeFilter(new[] { EConfigType.Custom, EConfigType.PolicyGroup, EConfigType.ProxyChain }, exclude: true); | ||||||
|  |         } | ||||||
|  |         selectWindow.AllowMultiSelect(true); | ||||||
|  |         if (selectWindow.ShowDialog() == true) | ||||||
|  |         { | ||||||
|  |             var profiles = await selectWindow.ProfileItems; | ||||||
|  |             ViewModel?.ChildItemsObs.AddRange(profiles); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void LstChild_SelectionChanged(object sender, System.Windows.Controls.SelectionChangedEventArgs e) | ||||||
|  |     { | ||||||
|  |         if (ViewModel != null) | ||||||
|  |         { | ||||||
|  |             ViewModel.SelectedChildren = lstChild.SelectedItems.Cast<ProfileItem>().ToList(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void MenuSelectAllChild_Click(object sender, RoutedEventArgs e) | ||||||
|  |     { | ||||||
|  |         lstChild.SelectAll(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -71,6 +71,14 @@ | ||||||
|                                     x:Name="menuAddCustomServer" |                                     x:Name="menuAddCustomServer" | ||||||
|                                     Height="{StaticResource MenuItemHeight}" |                                     Height="{StaticResource MenuItemHeight}" | ||||||
|                                     Header="{x:Static resx:ResUI.menuAddCustomServer}" /> |                                     Header="{x:Static resx:ResUI.menuAddCustomServer}" /> | ||||||
|  |                                 <MenuItem | ||||||
|  |                                     x:Name="menuAddPolicyGroupServer" | ||||||
|  |                                     Height="{StaticResource MenuItemHeight}" | ||||||
|  |                                     Header="{x:Static resx:ResUI.menuAddPolicyGroupServer}" /> | ||||||
|  |                                 <MenuItem | ||||||
|  |                                     x:Name="menuAddProxyChainServer" | ||||||
|  |                                     Height="{StaticResource MenuItemHeight}" | ||||||
|  |                                     Header="{x:Static resx:ResUI.menuAddProxyChainServer}" /> | ||||||
|                                 <Separator Margin="-40,5" /> |                                 <Separator Margin="-40,5" /> | ||||||
|                                 <MenuItem |                                 <MenuItem | ||||||
|                                     x:Name="menuAddVmessServer" |                                     x:Name="menuAddVmessServer" | ||||||
|  |  | ||||||
|  | @ -79,6 +79,8 @@ public partial class MainWindow | ||||||
|             this.BindCommand(ViewModel, vm => vm.AddWireguardServerCmd, v => v.menuAddWireguardServer).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.AddWireguardServerCmd, v => v.menuAddWireguardServer).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.AddAnytlsServerCmd, v => v.menuAddAnytlsServer).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.AddAnytlsServerCmd, v => v.menuAddAnytlsServer).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.AddCustomServerCmd, v => v.menuAddCustomServer).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.AddCustomServerCmd, v => v.menuAddCustomServer).DisposeWith(disposables); | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.AddPolicyGroupServerCmd, v => v.menuAddPolicyGroupServer).DisposeWith(disposables); | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.AddProxyChainServerCmd, v => v.menuAddProxyChainServer).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.AddServerViaClipboardCmd, v => v.menuAddServerViaClipboard).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.AddServerViaClipboardCmd, v => v.menuAddServerViaClipboard).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.AddServerViaScanCmd, v => v.menuAddServerViaScan).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.AddServerViaScanCmd, v => v.menuAddServerViaScan).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.AddServerViaImageCmd, v => v.menuAddServerViaImage).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.AddServerViaImageCmd, v => v.menuAddServerViaImage).DisposeWith(disposables); | ||||||
|  | @ -195,6 +197,11 @@ public partial class MainWindow | ||||||
|                     return false; |                     return false; | ||||||
|                 return (new AddServer2Window((ProfileItem)obj)).ShowDialog() ?? false; |                 return (new AddServer2Window((ProfileItem)obj)).ShowDialog() ?? false; | ||||||
| 
 | 
 | ||||||
|  |             case EViewAction.AddGroupServerWindow: | ||||||
|  |                 if (obj is null) | ||||||
|  |                     return false; | ||||||
|  |                 return (new AddGroupServerWindow((ProfileItem)obj)).ShowDialog() ?? false; | ||||||
|  | 
 | ||||||
|             case EViewAction.DNSSettingWindow: |             case EViewAction.DNSSettingWindow: | ||||||
|                 return (new DNSSettingWindow().ShowDialog() ?? false); |                 return (new DNSSettingWindow().ShowDialog() ?? false); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -48,18 +48,25 @@ public partial class MsgView | ||||||
|     private void ShowMsg(object msg) |     private void ShowMsg(object msg) | ||||||
|     { |     { | ||||||
|         txtMsg.BeginChange(); |         txtMsg.BeginChange(); | ||||||
|         txtMsg.Text = msg.ToString(); | 
 | ||||||
|  |         if (txtMsg.LineCount > ViewModel?.NumMaxMsg) | ||||||
|  |         { | ||||||
|  |             ClearMsg(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         txtMsg.AppendText(msg.ToString()); | ||||||
|         if (togScrollToEnd.IsChecked ?? true) |         if (togScrollToEnd.IsChecked ?? true) | ||||||
|         { |         { | ||||||
|             txtMsg.ScrollToEnd(); |             txtMsg.ScrollToEnd(); | ||||||
|         } |         } | ||||||
|  | 
 | ||||||
|         txtMsg.EndChange(); |         txtMsg.EndChange(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public void ClearMsg() |     public void ClearMsg() | ||||||
|     { |     { | ||||||
|         ViewModel?.ClearMsg(); |  | ||||||
|         txtMsg.Clear(); |         txtMsg.Clear(); | ||||||
|  |         txtMsg.AppendText("----- Message cleared -----\n"); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private void menuMsgViewSelectAll_Click(object sender, System.Windows.RoutedEventArgs e) |     private void menuMsgViewSelectAll_Click(object sender, System.Windows.RoutedEventArgs e) | ||||||
|  |  | ||||||
|  | @ -125,28 +125,32 @@ | ||||||
|                             Height="{StaticResource MenuItemHeight}" |                             Height="{StaticResource MenuItemHeight}" | ||||||
|                             Header="{x:Static resx:ResUI.menuShareServer}" /> |                             Header="{x:Static resx:ResUI.menuShareServer}" /> | ||||||
|                         <Separator /> |                         <Separator /> | ||||||
|                         <MenuItem Header="{x:Static resx:ResUI.menuSetDefaultMultipleServer}"> |                         <MenuItem Header="{x:Static resx:ResUI.menuGenGroupMultipleServer}"> | ||||||
|                             <MenuItem |                             <MenuItem | ||||||
|                                 x:Name="menuSetDefaultMultipleServerXrayRandom" |                                 x:Name="menuGenGroupMultipleServerXrayRandom" | ||||||
|                                 Height="{StaticResource MenuItemHeight}" |                                 Height="{StaticResource MenuItemHeight}" | ||||||
|                                 Header="{x:Static resx:ResUI.menuSetDefaultMultipleServerXrayRandom}" /> |                                 Header="{x:Static resx:ResUI.menuGenGroupMultipleServerXrayRandom}" /> | ||||||
|                             <MenuItem |                             <MenuItem | ||||||
|                                 x:Name="menuSetDefaultMultipleServerXrayRoundRobin" |                                 x:Name="menuGenGroupMultipleServerXrayRoundRobin" | ||||||
|                                 Height="{StaticResource MenuItemHeight}" |                                 Height="{StaticResource MenuItemHeight}" | ||||||
|                                 Header="{x:Static resx:ResUI.menuSetDefaultMultipleServerXrayRoundRobin}" /> |                                 Header="{x:Static resx:ResUI.menuGenGroupMultipleServerXrayRoundRobin}" /> | ||||||
|                             <MenuItem |                             <MenuItem | ||||||
|                                 x:Name="menuSetDefaultMultipleServerXrayLeastPing" |                                 x:Name="menuGenGroupMultipleServerXrayLeastPing" | ||||||
|                                 Height="{StaticResource MenuItemHeight}" |                                 Height="{StaticResource MenuItemHeight}" | ||||||
|                                 Header="{x:Static resx:ResUI.menuSetDefaultMultipleServerXrayLeastPing}" /> |                                 Header="{x:Static resx:ResUI.menuGenGroupMultipleServerXrayLeastPing}" /> | ||||||
|                             <MenuItem |                             <MenuItem | ||||||
|                                 x:Name="menuSetDefaultMultipleServerXrayLeastLoad" |                                 x:Name="menuGenGroupMultipleServerXrayLeastLoad" | ||||||
|                                 Height="{StaticResource MenuItemHeight}" |                                 Height="{StaticResource MenuItemHeight}" | ||||||
|                                 Header="{x:Static resx:ResUI.menuSetDefaultMultipleServerXrayLeastLoad}" /> |                                 Header="{x:Static resx:ResUI.menuGenGroupMultipleServerXrayLeastLoad}" /> | ||||||
|                             <Separator /> |                             <Separator /> | ||||||
|                             <MenuItem |                             <MenuItem | ||||||
|                                 x:Name="menuSetDefaultMultipleServerSingBoxLeastPing" |                                 x:Name="menuGenGroupMultipleServerSingBoxLeastPing" | ||||||
|                                 Height="{StaticResource MenuItemHeight}" |                                 Height="{StaticResource MenuItemHeight}" | ||||||
|                                 Header="{x:Static resx:ResUI.menuSetDefaultMultipleServerSingBoxLeastPing}" /> |                                 Header="{x:Static resx:ResUI.menuGenGroupMultipleServerSingBoxLeastPing}" /> | ||||||
|  |                             <MenuItem | ||||||
|  |                                 x:Name="menuGenGroupMultipleServerSingBoxFallback" | ||||||
|  |                                 Height="{StaticResource MenuItemHeight}" | ||||||
|  |                                 Header="{x:Static resx:ResUI.menuGenGroupMultipleServerSingBoxFallback}" /> | ||||||
|                         </MenuItem> |                         </MenuItem> | ||||||
|                         <Separator /> |                         <Separator /> | ||||||
|                         <MenuItem |                         <MenuItem | ||||||
|  |  | ||||||
|  | @ -60,11 +60,12 @@ public partial class ProfilesView | ||||||
|             this.BindCommand(ViewModel, vm => vm.CopyServerCmd, v => v.menuCopyServer).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.CopyServerCmd, v => v.menuCopyServer).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.SetDefaultServerCmd, v => v.menuSetDefaultServer).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.SetDefaultServerCmd, v => v.menuSetDefaultServer).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.ShareServerCmd, v => v.menuShareServer).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.ShareServerCmd, v => v.menuShareServer).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.SetDefaultMultipleServerXrayRandomCmd, v => v.menuSetDefaultMultipleServerXrayRandom).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.GenGroupMultipleServerXrayRandomCmd, v => v.menuGenGroupMultipleServerXrayRandom).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.SetDefaultMultipleServerXrayRoundRobinCmd, v => v.menuSetDefaultMultipleServerXrayRoundRobin).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.GenGroupMultipleServerXrayRoundRobinCmd, v => v.menuGenGroupMultipleServerXrayRoundRobin).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.SetDefaultMultipleServerXrayLeastPingCmd, v => v.menuSetDefaultMultipleServerXrayLeastPing).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.GenGroupMultipleServerXrayLeastPingCmd, v => v.menuGenGroupMultipleServerXrayLeastPing).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.SetDefaultMultipleServerXrayLeastLoadCmd, v => v.menuSetDefaultMultipleServerXrayLeastLoad).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.GenGroupMultipleServerXrayLeastLoadCmd, v => v.menuGenGroupMultipleServerXrayLeastLoad).DisposeWith(disposables); | ||||||
|             this.BindCommand(ViewModel, vm => vm.SetDefaultMultipleServerSingBoxLeastPingCmd, v => v.menuSetDefaultMultipleServerSingBoxLeastPing).DisposeWith(disposables); |             this.BindCommand(ViewModel, vm => vm.GenGroupMultipleServerSingBoxLeastPingCmd, v => v.menuGenGroupMultipleServerSingBoxLeastPing).DisposeWith(disposables); | ||||||
|  |             this.BindCommand(ViewModel, vm => vm.GenGroupMultipleServerSingBoxFallbackCmd, v => v.menuGenGroupMultipleServerSingBoxFallback).DisposeWith(disposables); | ||||||
| 
 | 
 | ||||||
|             //servers move |             //servers move | ||||||
|             this.OneWayBind(ViewModel, vm => vm.SubItems, v => v.cmbMoveToGroup.ItemsSource).DisposeWith(disposables); |             this.OneWayBind(ViewModel, vm => vm.SubItems, v => v.cmbMoveToGroup.ItemsSource).DisposeWith(disposables); | ||||||
|  | @ -148,6 +149,11 @@ public partial class ProfilesView | ||||||
|                     return false; |                     return false; | ||||||
|                 return (new AddServer2Window((ProfileItem)obj)).ShowDialog() ?? false; |                 return (new AddServer2Window((ProfileItem)obj)).ShowDialog() ?? false; | ||||||
| 
 | 
 | ||||||
|  |             case EViewAction.AddGroupServerWindow: | ||||||
|  |                 if (obj is null) | ||||||
|  |                     return false; | ||||||
|  |                 return (new AddGroupServerWindow((ProfileItem)obj)).ShowDialog() ?? false; | ||||||
|  | 
 | ||||||
|             case EViewAction.ShareServer: |             case EViewAction.ShareServer: | ||||||
|                 if (obj is null) |                 if (obj is null) | ||||||
|                     return false; |                     return false; | ||||||
|  |  | ||||||
|  | @ -57,7 +57,7 @@ public partial class SubEditWindow | ||||||
|     private async void BtnSelectPrevProfile_Click(object sender, RoutedEventArgs e) |     private async void BtnSelectPrevProfile_Click(object sender, RoutedEventArgs e) | ||||||
|     { |     { | ||||||
|         var selectWindow = new ProfilesSelectWindow(); |         var selectWindow = new ProfilesSelectWindow(); | ||||||
|         selectWindow.SetConfigTypeFilter(new[] { EConfigType.Custom }, exclude: true); |         selectWindow.SetConfigTypeFilter(new[] { EConfigType.Custom, EConfigType.PolicyGroup, EConfigType.ProxyChain }, exclude: true); | ||||||
|         if (selectWindow.ShowDialog() == true) |         if (selectWindow.ShowDialog() == true) | ||||||
|         { |         { | ||||||
|             var profile = await selectWindow.ProfileItem; |             var profile = await selectWindow.ProfileItem; | ||||||
|  | @ -71,7 +71,7 @@ public partial class SubEditWindow | ||||||
|     private async void BtnSelectNextProfile_Click(object sender, RoutedEventArgs e) |     private async void BtnSelectNextProfile_Click(object sender, RoutedEventArgs e) | ||||||
|     { |     { | ||||||
|         var selectWindow = new ProfilesSelectWindow(); |         var selectWindow = new ProfilesSelectWindow(); | ||||||
|         selectWindow.SetConfigTypeFilter(new[] { EConfigType.Custom }, exclude: true); |         selectWindow.SetConfigTypeFilter(new[] { EConfigType.Custom, EConfigType.PolicyGroup, EConfigType.ProxyChain }, exclude: true); | ||||||
|         if (selectWindow.ShowDialog() == true) |         if (selectWindow.ShowDialog() == true) | ||||||
|         { |         { | ||||||
|             var profile = await selectWindow.ProfileItem; |             var profile = await selectWindow.ProfileItem; | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue