mirror of
				https://github.com/2dust/v2rayN.git
				synced 2025-10-26 02:04:40 +00:00 
			
		
		
		
	Compare commits
	
		
			No commits in common. "471bc0f65dc4c142f06612bfd2ab12616dc9e790" and "e915726c52fd2f1dbd610d3b818f0b30963f2ca0" have entirely different histories.
		
	
	
		
			471bc0f65d
			...
			e915726c52
		
	
		
					 19 changed files with 411 additions and 95 deletions
				
			
		|  | @ -1,5 +1,5 @@ | ||||||
| # v2rayN | # v2rayN | ||||||
| A GUI client for Windows and Linux, support [Xray core](https://github.com/XTLS/Xray-core)  and [others](https://github.com/2dust/v2rayN/wiki/List-of-supported-cores) | A GUI client for Windows and Linux, support [Xray core](https://github.com/XTLS/Xray-core) and [v2fly core](https://github.com/v2fly/v2ray-core) and [others](https://github.com/2dust/v2rayN/wiki/List-of-supported-cores) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| [](https://github.com/2dust/v2rayN/commits/master) | [](https://github.com/2dust/v2rayN/commits/master) | ||||||
|  |  | ||||||
|  | @ -211,12 +211,12 @@ | ||||||
| 
 | 
 | ||||||
|         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>().Where(it => it.Locked == false).OrderBy(t => t.Sort).ToListAsync(); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         public async Task<RoutingItem?> GetRoutingItem(string id) |         public async Task<RoutingItem?> GetRoutingItem(string id) | ||||||
|         { |         { | ||||||
|             return await SQLiteHelper.Instance.TableAsync<RoutingItem>().FirstOrDefaultAsync(it => it.Id == id); |             return await SQLiteHelper.Instance.TableAsync<RoutingItem>().FirstOrDefaultAsync(it => it.Locked == false && it.Id == id); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         public async Task<List<DNSItem>?> DNSItems() |         public async Task<List<DNSItem>?> DNSItems() | ||||||
|  |  | ||||||
|  | @ -65,8 +65,10 @@ namespace ServiceLib.Handler | ||||||
|                     config.Inbound[0].Protocol = EInboundProtocol.socks.ToString(); |                     config.Inbound[0].Protocol = EInboundProtocol.socks.ToString(); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| 
 |             config.RoutingBasicItem ??= new() | ||||||
|             config.RoutingBasicItem ??= new(); |             { | ||||||
|  |                 EnableRoutingAdvanced = true | ||||||
|  |             }; | ||||||
| 
 | 
 | ||||||
|             if (Utils.IsNullOrEmpty(config.RoutingBasicItem.DomainStrategy)) |             if (Utils.IsNullOrEmpty(config.RoutingBasicItem.DomainStrategy)) | ||||||
|             { |             { | ||||||
|  | @ -1296,20 +1298,6 @@ namespace ServiceLib.Handler | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             //Keep the last traffic statistics |  | ||||||
|             if (lstOriSub != null) |  | ||||||
|             { |  | ||||||
|                 var lstSub = await AppHandler.Instance.ProfileItems(subid); |  | ||||||
|                 foreach (var item in lstSub) |  | ||||||
|                 { |  | ||||||
|                     var existItem = lstOriSub?.FirstOrDefault(t => config.UiItem.EnableUpdateSubOnlyRemarksExist ? t.Remarks == item.Remarks : CompareProfileItem(t, item, true)); |  | ||||||
|                     if (existItem != null) |  | ||||||
|                     { |  | ||||||
|                         await StatisticsHandler.Instance.CloneServerStatItem(existItem.IndexId, item.IndexId); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return counter; |             return counter; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | @ -1612,7 +1600,7 @@ namespace ServiceLib.Handler | ||||||
|             var item = await AppHandler.Instance.GetRoutingItem(config.RoutingBasicItem.RoutingIndexId); |             var item = await AppHandler.Instance.GetRoutingItem(config.RoutingBasicItem.RoutingIndexId); | ||||||
|             if (item is null) |             if (item is null) | ||||||
|             { |             { | ||||||
|                 var item2 = await SQLiteHelper.Instance.TableAsync<RoutingItem>().FirstOrDefaultAsync(); |                 var item2 = await SQLiteHelper.Instance.TableAsync<RoutingItem>().FirstOrDefaultAsync(t => t.Locked == false); | ||||||
|                 await SetDefaultRouting(config, item2); |                 await SetDefaultRouting(config, item2); | ||||||
|                 return item2; |                 return item2; | ||||||
|             } |             } | ||||||
|  | @ -1686,15 +1674,6 @@ namespace ServiceLib.Handler | ||||||
|         { |         { | ||||||
|             var ver = "V3-"; |             var ver = "V3-"; | ||||||
|             var items = await AppHandler.Instance.RoutingItems(); |             var items = await AppHandler.Instance.RoutingItems(); | ||||||
| 
 |  | ||||||
|             //TODO Temporary code to be removed later |  | ||||||
|             var lockItem = items?.FirstOrDefault(t => t.Locked == true); |  | ||||||
|             if (lockItem != null) |  | ||||||
|             { |  | ||||||
|                 await ConfigHandler.RemoveRoutingItem(lockItem); |  | ||||||
|                 items = await AppHandler.Instance.RoutingItems(); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (!blImportAdvancedRules && items.Where(t => t.Remarks.StartsWith(ver)).ToList().Count > 0) |             if (!blImportAdvancedRules && items.Where(t => t.Remarks.StartsWith(ver)).ToList().Count > 0) | ||||||
|             { |             { | ||||||
|                 return 0; |                 return 0; | ||||||
|  | @ -1735,6 +1714,11 @@ namespace ServiceLib.Handler | ||||||
|             return 0; |             return 0; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         public static async Task<RoutingItem?> GetLockedRoutingItem(Config config) | ||||||
|  |         { | ||||||
|  |             return await SQLiteHelper.Instance.TableAsync<RoutingItem>().FirstOrDefaultAsync(it => it.Locked == true); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         public static async Task RemoveRoutingItem(RoutingItem routingItem) |         public static async Task RemoveRoutingItem(RoutingItem routingItem) | ||||||
|         { |         { | ||||||
|             await SQLiteHelper.Instance.DeleteAsync(routingItem); |             await SQLiteHelper.Instance.DeleteAsync(routingItem); | ||||||
|  |  | ||||||
|  | @ -65,25 +65,6 @@ | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         public async Task CloneServerStatItem(string indexId, string toIndexId) |  | ||||||
|         { |  | ||||||
|             if (_lstServerStat == null) |  | ||||||
|             { |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             var stat = _lstServerStat.FirstOrDefault(t => t.IndexId == indexId); |  | ||||||
|             if (stat == null) |  | ||||||
|             { |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             var toStat = JsonUtils.DeepCopy(stat); |  | ||||||
|             toStat.IndexId = toIndexId; |  | ||||||
|             await SQLiteHelper.Instance.ReplaceAsync(toStat); |  | ||||||
|             _lstServerStat.Add(toStat); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private async Task InitData() |         private async Task InitData() | ||||||
|         { |         { | ||||||
|             await SQLiteHelper.Instance.ExecuteAsync($"delete from ServerStatItem where indexId not in ( select indexId from ProfileItem )"); |             await SQLiteHelper.Instance.ExecuteAsync($"delete from ServerStatItem where indexId not in ( select indexId from ProfileItem )"); | ||||||
|  |  | ||||||
|  | @ -180,6 +180,7 @@ | ||||||
|         public string DomainStrategy4Singbox { get; set; } |         public string DomainStrategy4Singbox { get; set; } | ||||||
|         public string DomainMatcher { get; set; } |         public string DomainMatcher { get; set; } | ||||||
|         public string RoutingIndexId { get; set; } |         public string RoutingIndexId { get; set; } | ||||||
|  |         public bool EnableRoutingAdvanced { get; set; } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     [Serializable] |     [Serializable] | ||||||
|  |  | ||||||
							
								
								
									
										18
									
								
								v2rayN/ServiceLib/Resx/ResUI.Designer.cs
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										18
									
								
								v2rayN/ServiceLib/Resx/ResUI.Designer.cs
									
									
									
										generated
									
									
									
								
							|  | @ -1365,6 +1365,24 @@ namespace ServiceLib.Resx { | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|          |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Basic Function 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string menuRoutingBasic { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("menuRoutingBasic", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         /// <summary> | ||||||
|  |         ///   查找类似 Import Basic Rules 的本地化字符串。 | ||||||
|  |         /// </summary> | ||||||
|  |         public static string menuRoutingBasicImportRules { | ||||||
|  |             get { | ||||||
|  |                 return ResourceManager.GetString("menuRoutingBasicImportRules", resourceCulture); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 RoutingRuleDetailsSetting 的本地化字符串。 |         ///   查找类似 RoutingRuleDetailsSetting 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|  |  | ||||||
|  | @ -847,6 +847,12 @@ | ||||||
|   <data name="menuRoutingAdvancedSetDefault" xml:space="preserve"> |   <data name="menuRoutingAdvancedSetDefault" xml:space="preserve"> | ||||||
|     <value>Set as active rule</value> |     <value>Set as active rule</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="menuRoutingBasic" xml:space="preserve"> | ||||||
|  |     <value>عملکرد پایه</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuRoutingBasicImportRules" xml:space="preserve"> | ||||||
|  |     <value>واردات قوانین اساسی</value> | ||||||
|  |   </data> | ||||||
|   <data name="TbdomainMatcher" xml:space="preserve"> |   <data name="TbdomainMatcher" xml:space="preserve"> | ||||||
|     <value>تطبیق دامنه</value> |     <value>تطبیق دامنه</value> | ||||||
|   </data> |   </data> | ||||||
|  |  | ||||||
|  | @ -850,6 +850,12 @@ | ||||||
|   <data name="menuRoutingAdvancedSetDefault" xml:space="preserve"> |   <data name="menuRoutingAdvancedSetDefault" xml:space="preserve"> | ||||||
|     <value>Set as active rule(Enter)</value> |     <value>Set as active rule(Enter)</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="menuRoutingBasic" xml:space="preserve"> | ||||||
|  |     <value>Basic Function</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuRoutingBasicImportRules" xml:space="preserve"> | ||||||
|  |     <value>Import Basic Rules</value> | ||||||
|  |   </data> | ||||||
|   <data name="TbdomainMatcher" xml:space="preserve"> |   <data name="TbdomainMatcher" xml:space="preserve"> | ||||||
|     <value>Domain Matcher</value> |     <value>Domain Matcher</value> | ||||||
|   </data> |   </data> | ||||||
|  |  | ||||||
|  | @ -856,6 +856,12 @@ | ||||||
|   <data name="menuRoutingAdvancedSetDefault" xml:space="preserve"> |   <data name="menuRoutingAdvancedSetDefault" xml:space="preserve"> | ||||||
|     <value>Установить как активное правило</value> |     <value>Установить как активное правило</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="menuRoutingBasic" xml:space="preserve"> | ||||||
|  |     <value>Основные функции</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuRoutingBasicImportRules" xml:space="preserve"> | ||||||
|  |     <value>Импорт основных правил</value> | ||||||
|  |   </data> | ||||||
|   <data name="TbdomainMatcher" xml:space="preserve"> |   <data name="TbdomainMatcher" xml:space="preserve"> | ||||||
|     <value>Сопоставитель доменов</value> |     <value>Сопоставитель доменов</value> | ||||||
|   </data> |   </data> | ||||||
|  |  | ||||||
|  | @ -850,6 +850,12 @@ | ||||||
|   <data name="menuRoutingAdvancedSetDefault" xml:space="preserve"> |   <data name="menuRoutingAdvancedSetDefault" xml:space="preserve"> | ||||||
|     <value>设为活动规则 (Enter)</value> |     <value>设为活动规则 (Enter)</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="menuRoutingBasic" xml:space="preserve"> | ||||||
|  |     <value>基础功能</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuRoutingBasicImportRules" xml:space="preserve"> | ||||||
|  |     <value>一键导入基础规则</value> | ||||||
|  |   </data> | ||||||
|   <data name="TbdomainMatcher" xml:space="preserve"> |   <data name="TbdomainMatcher" xml:space="preserve"> | ||||||
|     <value>域名匹配算法</value> |     <value>域名匹配算法</value> | ||||||
|   </data> |   </data> | ||||||
|  |  | ||||||
|  | @ -850,6 +850,12 @@ | ||||||
|   <data name="menuRoutingAdvancedSetDefault" xml:space="preserve"> |   <data name="menuRoutingAdvancedSetDefault" xml:space="preserve"> | ||||||
|     <value>設為活動規則 (Enter)</value> |     <value>設為活動規則 (Enter)</value> | ||||||
|   </data> |   </data> | ||||||
|  |   <data name="menuRoutingBasic" xml:space="preserve"> | ||||||
|  |     <value>基礎功能</value> | ||||||
|  |   </data> | ||||||
|  |   <data name="menuRoutingBasicImportRules" xml:space="preserve"> | ||||||
|  |     <value>一鍵匯入基礎規則</value> | ||||||
|  |   </data> | ||||||
|   <data name="TbdomainMatcher" xml:space="preserve"> |   <data name="TbdomainMatcher" xml:space="preserve"> | ||||||
|     <value>域名匹配演算法</value> |     <value>域名匹配演算法</value> | ||||||
|   </data> |   </data> | ||||||
|  |  | ||||||
|  | @ -90,8 +90,8 @@ namespace ServiceLib.Services.CoreConfig | ||||||
| 
 | 
 | ||||||
|                 ret.Msg = ResUI.InitialConfiguration; |                 ret.Msg = ResUI.InitialConfiguration; | ||||||
| 
 | 
 | ||||||
|                 var result = Utils.GetEmbedText(Global.SingboxSampleClient); |                 string result = Utils.GetEmbedText(Global.SingboxSampleClient); | ||||||
|                 var txtOutbound = Utils.GetEmbedText(Global.SingboxSampleOutbound); |                 string txtOutbound = Utils.GetEmbedText(Global.SingboxSampleOutbound); | ||||||
|                 if (Utils.IsNullOrEmpty(result) || txtOutbound.IsNullOrEmpty()) |                 if (Utils.IsNullOrEmpty(result) || txtOutbound.IsNullOrEmpty()) | ||||||
|                 { |                 { | ||||||
|                     ret.Msg = ResUI.FailedGetDefaultConfiguration; |                     ret.Msg = ResUI.FailedGetDefaultConfiguration; | ||||||
|  | @ -119,10 +119,10 @@ namespace ServiceLib.Services.CoreConfig | ||||||
| 
 | 
 | ||||||
|                 await GenLog(singboxConfig); |                 await GenLog(singboxConfig); | ||||||
|                 //GenDns(new(), singboxConfig); |                 //GenDns(new(), singboxConfig); | ||||||
|                 singboxConfig.inbounds.Clear(); |                 singboxConfig.inbounds.Clear(); // Remove "proxy" service for speedtest, avoiding port conflicts. | ||||||
|                 singboxConfig.outbounds.RemoveAt(0); |                 singboxConfig.outbounds.RemoveAt(0); | ||||||
| 
 | 
 | ||||||
|                 var httpPort = AppHandler.Instance.GetLocalPort(EInboundProtocol.speedtest); |                 int httpPort = AppHandler.Instance.GetLocalPort(EInboundProtocol.speedtest); | ||||||
| 
 | 
 | ||||||
|                 foreach (var it in selecteds) |                 foreach (var it in selecteds) | ||||||
|                 { |                 { | ||||||
|  | @ -499,10 +499,13 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                     inbound.sniff_override_destination = _config.Inbound[0].RouteOnly ? false : _config.Inbound[0].SniffingEnabled; |                     inbound.sniff_override_destination = _config.Inbound[0].RouteOnly ? false : _config.Inbound[0].SniffingEnabled; | ||||||
|                     inbound.domain_strategy = Utils.IsNullOrEmpty(_config.RoutingBasicItem.DomainStrategy4Singbox) ? null : _config.RoutingBasicItem.DomainStrategy4Singbox; |                     inbound.domain_strategy = Utils.IsNullOrEmpty(_config.RoutingBasicItem.DomainStrategy4Singbox) ? null : _config.RoutingBasicItem.DomainStrategy4Singbox; | ||||||
| 
 | 
 | ||||||
|                     var routing = await ConfigHandler.GetDefaultRouting(_config); |                     if (_config.RoutingBasicItem.EnableRoutingAdvanced) | ||||||
|                     if (Utils.IsNotEmpty(routing.DomainStrategy4Singbox)) |  | ||||||
|                     { |                     { | ||||||
|                         inbound.domain_strategy = routing.DomainStrategy4Singbox; |                         var routing = await ConfigHandler.GetDefaultRouting(_config); | ||||||
|  |                         if (Utils.IsNotEmpty(routing.DomainStrategy4Singbox)) | ||||||
|  |                         { | ||||||
|  |                             inbound.domain_strategy = routing.DomainStrategy4Singbox; | ||||||
|  |                         } | ||||||
|                     } |                     } | ||||||
| 
 | 
 | ||||||
|                     //http |                     //http | ||||||
|  | @ -955,13 +958,28 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                     }); |                     }); | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 var routing = await ConfigHandler.GetDefaultRouting(_config); |                 if (_config.RoutingBasicItem.EnableRoutingAdvanced) | ||||||
|                 if (routing != null) |  | ||||||
|                 { |                 { | ||||||
|                     var rules = JsonUtils.Deserialize<List<RulesItem>>(routing.RuleSet); |                     var routing = await ConfigHandler.GetDefaultRouting(_config); | ||||||
|                     foreach (var item in rules ?? []) |                     if (routing != null) | ||||||
|                     { |                     { | ||||||
|                         if (item.Enabled) |                         var rules = JsonUtils.Deserialize<List<RulesItem>>(routing.RuleSet); | ||||||
|  |                         foreach (var item in rules ?? []) | ||||||
|  |                         { | ||||||
|  |                             if (item.Enabled) | ||||||
|  |                             { | ||||||
|  |                                 await GenRoutingUserRule(item, singboxConfig.route.rules); | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 else | ||||||
|  |                 { | ||||||
|  |                     var lockedItem = await ConfigHandler.GetLockedRoutingItem(_config); | ||||||
|  |                     if (lockedItem != null) | ||||||
|  |                     { | ||||||
|  |                         var rules = JsonUtils.Deserialize<List<RulesItem>>(lockedItem.RuleSet); | ||||||
|  |                         foreach (var item in rules ?? []) | ||||||
|                         { |                         { | ||||||
|                             await GenRoutingUserRule(item, singboxConfig.route.rules); |                             await GenRoutingUserRule(item, singboxConfig.route.rules); | ||||||
|                         } |                         } | ||||||
|  | @ -1316,18 +1334,20 @@ namespace ServiceLib.Services.CoreConfig | ||||||
| 
 | 
 | ||||||
|             //load custom ruleset file |             //load custom ruleset file | ||||||
|             List<Ruleset4Sbox> customRulesets = []; |             List<Ruleset4Sbox> customRulesets = []; | ||||||
| 
 |             if (_config.RoutingBasicItem.EnableRoutingAdvanced) | ||||||
|             var routing = await ConfigHandler.GetDefaultRouting(_config); |  | ||||||
|             if (Utils.IsNotEmpty(routing.CustomRulesetPath4Singbox)) |  | ||||||
|             { |             { | ||||||
|                 var result = Utils.LoadResource(routing.CustomRulesetPath4Singbox); |                 var routing = await ConfigHandler.GetDefaultRouting(_config); | ||||||
|                 if (Utils.IsNotEmpty(result)) |                 if (Utils.IsNotEmpty(routing.CustomRulesetPath4Singbox)) | ||||||
|                 { |                 { | ||||||
|                     customRulesets = (JsonUtils.Deserialize<List<Ruleset4Sbox>>(result) ?? []) |                     var result = Utils.LoadResource(routing.CustomRulesetPath4Singbox); | ||||||
|                         .Where(t => t.tag != null) |                     if (Utils.IsNotEmpty(result)) | ||||||
|                         .Where(t => t.type != null) |                     { | ||||||
|                         .Where(t => t.format != null) |                         customRulesets = (JsonUtils.Deserialize<List<Ruleset4Sbox>>(result) ?? []) | ||||||
|                         .ToList(); |                             .Where(t => t.tag != null) | ||||||
|  |                             .Where(t => t.type != null) | ||||||
|  |                             .Where(t => t.format != null) | ||||||
|  |                             .ToList(); | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -216,8 +216,8 @@ namespace ServiceLib.Services.CoreConfig | ||||||
| 
 | 
 | ||||||
|                 ret.Msg = ResUI.InitialConfiguration; |                 ret.Msg = ResUI.InitialConfiguration; | ||||||
| 
 | 
 | ||||||
|                 var result = Utils.GetEmbedText(Global.V2raySampleClient); |                 string result = Utils.GetEmbedText(Global.V2raySampleClient); | ||||||
|                 var txtOutbound = Utils.GetEmbedText(Global.V2raySampleOutbound); |                 string txtOutbound = Utils.GetEmbedText(Global.V2raySampleOutbound); | ||||||
|                 if (Utils.IsNullOrEmpty(result) || txtOutbound.IsNullOrEmpty()) |                 if (Utils.IsNullOrEmpty(result) || txtOutbound.IsNullOrEmpty()) | ||||||
|                 { |                 { | ||||||
|                     ret.Msg = ResUI.FailedGetDefaultConfiguration; |                     ret.Msg = ResUI.FailedGetDefaultConfiguration; | ||||||
|  | @ -244,11 +244,10 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 await GenLog(v2rayConfig); |                 await GenLog(v2rayConfig); | ||||||
|                 v2rayConfig.inbounds.Clear(); |                 v2rayConfig.inbounds.Clear(); // Remove "proxy" service for speedtest, avoiding port conflicts. | ||||||
|                 v2rayConfig.outbounds.Clear(); |                 v2rayConfig.outbounds.RemoveAt(0); | ||||||
|                 v2rayConfig.routing.rules.Clear(); |  | ||||||
| 
 | 
 | ||||||
|                 var httpPort = AppHandler.Instance.GetLocalPort(EInboundProtocol.speedtest); |                 int httpPort = AppHandler.Instance.GetLocalPort(EInboundProtocol.speedtest); | ||||||
| 
 | 
 | ||||||
|                 foreach (var it in selecteds) |                 foreach (var it in selecteds) | ||||||
|                 { |                 { | ||||||
|  | @ -271,7 +270,7 @@ namespace ServiceLib.Services.CoreConfig | ||||||
| 
 | 
 | ||||||
|                     //find unused port |                     //find unused port | ||||||
|                     var port = httpPort; |                     var port = httpPort; | ||||||
|                     for (var k = httpPort; k < Global.MaxPort; k++) |                     for (int k = httpPort; k < Global.MaxPort; k++) | ||||||
|                     { |                     { | ||||||
|                         if (lstIpEndPoints?.FindIndex(_it => _it.Port == k) >= 0) |                         if (lstIpEndPoints?.FindIndex(_it => _it.Port == k) >= 0) | ||||||
|                         { |                         { | ||||||
|  | @ -295,6 +294,16 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                     it.Port = port; |                     it.Port = port; | ||||||
|                     it.AllowTest = true; |                     it.AllowTest = true; | ||||||
| 
 | 
 | ||||||
|  |                     //inbound | ||||||
|  |                     Inbounds4Ray inbound = new() | ||||||
|  |                     { | ||||||
|  |                         listen = Global.Loopback, | ||||||
|  |                         port = port, | ||||||
|  |                         protocol = EInboundProtocol.http.ToString(), | ||||||
|  |                     }; | ||||||
|  |                     inbound.tag = inbound.protocol + inbound.port.ToString(); | ||||||
|  |                     v2rayConfig.inbounds.Add(inbound); | ||||||
|  | 
 | ||||||
|                     //outbound |                     //outbound | ||||||
|                     if (item is null) |                     if (item is null) | ||||||
|                     { |                     { | ||||||
|  | @ -317,16 +326,6 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                         continue; |                         continue; | ||||||
|                     } |                     } | ||||||
| 
 | 
 | ||||||
|                     //inbound |  | ||||||
|                     Inbounds4Ray inbound = new() |  | ||||||
|                     { |  | ||||||
|                         listen = Global.Loopback, |  | ||||||
|                         port = port, |  | ||||||
|                         protocol = EInboundProtocol.http.ToString(), |  | ||||||
|                     }; |  | ||||||
|                     inbound.tag = inbound.protocol + inbound.port.ToString(); |  | ||||||
|                     v2rayConfig.inbounds.Add(inbound); |  | ||||||
| 
 |  | ||||||
|                     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(); | ||||||
|  | @ -467,17 +466,33 @@ namespace ServiceLib.Services.CoreConfig | ||||||
|                     v2rayConfig.routing.domainStrategy = _config.RoutingBasicItem.DomainStrategy; |                     v2rayConfig.routing.domainStrategy = _config.RoutingBasicItem.DomainStrategy; | ||||||
|                     v2rayConfig.routing.domainMatcher = Utils.IsNullOrEmpty(_config.RoutingBasicItem.DomainMatcher) ? null : _config.RoutingBasicItem.DomainMatcher; |                     v2rayConfig.routing.domainMatcher = Utils.IsNullOrEmpty(_config.RoutingBasicItem.DomainMatcher) ? null : _config.RoutingBasicItem.DomainMatcher; | ||||||
| 
 | 
 | ||||||
|                     var routing = await ConfigHandler.GetDefaultRouting(_config); |                     if (_config.RoutingBasicItem.EnableRoutingAdvanced) | ||||||
|                     if (routing != null) |  | ||||||
|                     { |                     { | ||||||
|                         if (Utils.IsNotEmpty(routing.DomainStrategy)) |                         var routing = await ConfigHandler.GetDefaultRouting(_config); | ||||||
|  |                         if (routing != null) | ||||||
|                         { |                         { | ||||||
|                             v2rayConfig.routing.domainStrategy = routing.DomainStrategy; |                             if (Utils.IsNotEmpty(routing.DomainStrategy)) | ||||||
|  |                             { | ||||||
|  |                                 v2rayConfig.routing.domainStrategy = routing.DomainStrategy; | ||||||
|  |                             } | ||||||
|  |                             var rules = JsonUtils.Deserialize<List<RulesItem>>(routing.RuleSet); | ||||||
|  |                             foreach (var item in rules) | ||||||
|  |                             { | ||||||
|  |                                 if (item.Enabled) | ||||||
|  |                                 { | ||||||
|  |                                     var item2 = JsonUtils.Deserialize<RulesItem4Ray>(JsonUtils.Serialize(item)); | ||||||
|  |                                     await GenRoutingUserRule(item2, v2rayConfig); | ||||||
|  |                                 } | ||||||
|  |                             } | ||||||
|                         } |                         } | ||||||
|                         var rules = JsonUtils.Deserialize<List<RulesItem>>(routing.RuleSet); |                     } | ||||||
|                         foreach (var item in rules) |                     else | ||||||
|  |                     { | ||||||
|  |                         var lockedItem = await ConfigHandler.GetLockedRoutingItem(_config); | ||||||
|  |                         if (lockedItem != null) | ||||||
|                         { |                         { | ||||||
|                             if (item.Enabled) |                             var rules = JsonUtils.Deserialize<List<RulesItem>>(lockedItem.RuleSet); | ||||||
|  |                             foreach (var item in rules) | ||||||
|                             { |                             { | ||||||
|                                 var item2 = JsonUtils.Deserialize<RulesItem4Ray>(JsonUtils.Serialize(item)); |                                 var item2 = JsonUtils.Deserialize<RulesItem4Ray>(JsonUtils.Serialize(item)); | ||||||
|                                 await GenRoutingUserRule(item2, v2rayConfig); |                                 await GenRoutingUserRule(item2, v2rayConfig); | ||||||
|  |  | ||||||
|  | @ -7,6 +7,9 @@ namespace ServiceLib.ViewModels | ||||||
| { | { | ||||||
|     public class RoutingSettingViewModel : MyReactiveObject |     public class RoutingSettingViewModel : MyReactiveObject | ||||||
|     { |     { | ||||||
|  |         private RoutingItem _lockedItem; | ||||||
|  |         private List<RulesItem> _lockedRules; | ||||||
|  | 
 | ||||||
|         #region Reactive |         #region Reactive | ||||||
| 
 | 
 | ||||||
|         private IObservableCollection<RoutingItemModel> _routingItems = new ObservableCollectionExtended<RoutingItemModel>(); |         private IObservableCollection<RoutingItemModel> _routingItems = new ObservableCollectionExtended<RoutingItemModel>(); | ||||||
|  | @ -17,6 +20,12 @@ namespace ServiceLib.ViewModels | ||||||
| 
 | 
 | ||||||
|         public IList<RoutingItemModel> SelectedSources { get; set; } |         public IList<RoutingItemModel> SelectedSources { get; set; } | ||||||
| 
 | 
 | ||||||
|  |         [Reactive] | ||||||
|  |         public bool enableRoutingAdvanced { get; set; } | ||||||
|  | 
 | ||||||
|  |         [Reactive] | ||||||
|  |         public bool enableRoutingBasic { get; set; } | ||||||
|  | 
 | ||||||
|         [Reactive] |         [Reactive] | ||||||
|         public string domainStrategy { get; set; } |         public string domainStrategy { get; set; } | ||||||
| 
 | 
 | ||||||
|  | @ -26,6 +35,25 @@ namespace ServiceLib.ViewModels | ||||||
|         [Reactive] |         [Reactive] | ||||||
|         public string domainStrategy4Singbox { get; set; } |         public string domainStrategy4Singbox { get; set; } | ||||||
| 
 | 
 | ||||||
|  |         [Reactive] | ||||||
|  |         public string ProxyDomain { get; set; } | ||||||
|  | 
 | ||||||
|  |         [Reactive] | ||||||
|  |         public string ProxyIP { get; set; } | ||||||
|  | 
 | ||||||
|  |         [Reactive] | ||||||
|  |         public string DirectDomain { get; set; } | ||||||
|  | 
 | ||||||
|  |         [Reactive] | ||||||
|  |         public string DirectIP { get; set; } | ||||||
|  | 
 | ||||||
|  |         [Reactive] | ||||||
|  |         public string BlockDomain { get; set; } | ||||||
|  | 
 | ||||||
|  |         [Reactive] | ||||||
|  |         public string BlockIP { get; set; } | ||||||
|  | 
 | ||||||
|  |         public ReactiveCommand<Unit, Unit> RoutingBasicImportRulesCmd { get; } | ||||||
|         public ReactiveCommand<Unit, Unit> RoutingAdvancedAddCmd { get; } |         public ReactiveCommand<Unit, Unit> RoutingAdvancedAddCmd { get; } | ||||||
|         public ReactiveCommand<Unit, Unit> RoutingAdvancedRemoveCmd { get; } |         public ReactiveCommand<Unit, Unit> RoutingAdvancedRemoveCmd { get; } | ||||||
|         public ReactiveCommand<Unit, Unit> RoutingAdvancedSetDefaultCmd { get; } |         public ReactiveCommand<Unit, Unit> RoutingAdvancedSetDefaultCmd { get; } | ||||||
|  | @ -45,6 +73,15 @@ namespace ServiceLib.ViewModels | ||||||
|                 x => x.SelectedSource, |                 x => x.SelectedSource, | ||||||
|                 selectedSource => selectedSource != null && !selectedSource.Remarks.IsNullOrEmpty()); |                 selectedSource => selectedSource != null && !selectedSource.Remarks.IsNullOrEmpty()); | ||||||
| 
 | 
 | ||||||
|  |             this.WhenAnyValue( | ||||||
|  |                     x => x.enableRoutingAdvanced) | ||||||
|  |                 .Subscribe(c => enableRoutingBasic = !enableRoutingAdvanced); | ||||||
|  | 
 | ||||||
|  |             RoutingBasicImportRulesCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|  |             { | ||||||
|  |                 await RoutingBasicImportRules(); | ||||||
|  |             }); | ||||||
|  | 
 | ||||||
|             RoutingAdvancedAddCmd = ReactiveCommand.CreateFromTask(async () => |             RoutingAdvancedAddCmd = ReactiveCommand.CreateFromTask(async () => | ||||||
|             { |             { | ||||||
|                 await RoutingAdvancedEditAsync(true); |                 await RoutingAdvancedEditAsync(true); | ||||||
|  | @ -74,14 +111,67 @@ namespace ServiceLib.ViewModels | ||||||
|         { |         { | ||||||
|             SelectedSource = new(); |             SelectedSource = new(); | ||||||
| 
 | 
 | ||||||
|  |             enableRoutingAdvanced = true;//TODO _config.RoutingBasicItem.EnableRoutingAdvanced; | ||||||
|             domainStrategy = _config.RoutingBasicItem.DomainStrategy; |             domainStrategy = _config.RoutingBasicItem.DomainStrategy; | ||||||
|             domainMatcher = _config.RoutingBasicItem.DomainMatcher; |             domainMatcher = _config.RoutingBasicItem.DomainMatcher; | ||||||
|             domainStrategy4Singbox = _config.RoutingBasicItem.DomainStrategy4Singbox; |             domainStrategy4Singbox = _config.RoutingBasicItem.DomainStrategy4Singbox; | ||||||
| 
 | 
 | ||||||
|             await ConfigHandler.InitBuiltinRouting(_config); |             await ConfigHandler.InitBuiltinRouting(_config); | ||||||
|             await RefreshRoutingItems(); |             await RefreshRoutingItems(); | ||||||
|  |             await BindingLockedData(); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         #region locked | ||||||
|  | 
 | ||||||
|  |         private async Task BindingLockedData() | ||||||
|  |         { | ||||||
|  |             _lockedItem = await ConfigHandler.GetLockedRoutingItem(_config); | ||||||
|  |             if (_lockedItem == null) | ||||||
|  |             { | ||||||
|  |                 _lockedItem = new RoutingItem() | ||||||
|  |                 { | ||||||
|  |                     Remarks = "locked", | ||||||
|  |                     Url = string.Empty, | ||||||
|  |                     Locked = true, | ||||||
|  |                 }; | ||||||
|  |                 await ConfigHandler.AddBatchRoutingRules(_lockedItem, Utils.GetEmbedText(Global.CustomRoutingFileName + "locked")); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (_lockedItem != null) | ||||||
|  |             { | ||||||
|  |                 _lockedRules = JsonUtils.Deserialize<List<RulesItem>>(_lockedItem.RuleSet); | ||||||
|  |                 ProxyDomain = Utils.List2String(_lockedRules[0].Domain, true); | ||||||
|  |                 ProxyIP = Utils.List2String(_lockedRules[0].Ip, true); | ||||||
|  | 
 | ||||||
|  |                 DirectDomain = Utils.List2String(_lockedRules[1].Domain, true); | ||||||
|  |                 DirectIP = Utils.List2String(_lockedRules[1].Ip, true); | ||||||
|  | 
 | ||||||
|  |                 BlockDomain = Utils.List2String(_lockedRules[2].Domain, true); | ||||||
|  |                 BlockIP = Utils.List2String(_lockedRules[2].Ip, true); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private async Task EndBindingLockedData() | ||||||
|  |         { | ||||||
|  |             if (_lockedItem != null) | ||||||
|  |             { | ||||||
|  |                 _lockedRules[0].Domain = Utils.String2List(Utils.Convert2Comma(ProxyDomain.TrimEx())); | ||||||
|  |                 _lockedRules[0].Ip = Utils.String2List(Utils.Convert2Comma(ProxyIP.TrimEx())); | ||||||
|  | 
 | ||||||
|  |                 _lockedRules[1].Domain = Utils.String2List(Utils.Convert2Comma(DirectDomain.TrimEx())); | ||||||
|  |                 _lockedRules[1].Ip = Utils.String2List(Utils.Convert2Comma(DirectIP.TrimEx())); | ||||||
|  | 
 | ||||||
|  |                 _lockedRules[2].Domain = Utils.String2List(Utils.Convert2Comma(BlockDomain.TrimEx())); | ||||||
|  |                 _lockedRules[2].Ip = Utils.String2List(Utils.Convert2Comma(BlockIP.TrimEx())); | ||||||
|  | 
 | ||||||
|  |                 _lockedItem.RuleSet = JsonUtils.Serialize(_lockedRules, false); | ||||||
|  | 
 | ||||||
|  |                 await ConfigHandler.SaveRoutingItem(_config, _lockedItem); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         #endregion locked | ||||||
|  | 
 | ||||||
|         #region Refresh Save |         #region Refresh Save | ||||||
| 
 | 
 | ||||||
|         public async Task RefreshRoutingItems() |         public async Task RefreshRoutingItems() | ||||||
|  | @ -115,9 +205,12 @@ namespace ServiceLib.ViewModels | ||||||
|         private async Task SaveRoutingAsync() |         private async Task SaveRoutingAsync() | ||||||
|         { |         { | ||||||
|             _config.RoutingBasicItem.DomainStrategy = domainStrategy; |             _config.RoutingBasicItem.DomainStrategy = domainStrategy; | ||||||
|  |             _config.RoutingBasicItem.EnableRoutingAdvanced = enableRoutingAdvanced; | ||||||
|             _config.RoutingBasicItem.DomainMatcher = domainMatcher; |             _config.RoutingBasicItem.DomainMatcher = domainMatcher; | ||||||
|             _config.RoutingBasicItem.DomainStrategy4Singbox = domainStrategy4Singbox; |             _config.RoutingBasicItem.DomainStrategy4Singbox = domainStrategy4Singbox; | ||||||
| 
 | 
 | ||||||
|  |             await EndBindingLockedData(); | ||||||
|  | 
 | ||||||
|             if (await ConfigHandler.SaveConfig(_config) == 0) |             if (await ConfigHandler.SaveConfig(_config) == 0) | ||||||
|             { |             { | ||||||
|                 NoticeHandler.Instance.Enqueue(ResUI.OperationSuccess); |                 NoticeHandler.Instance.Enqueue(ResUI.OperationSuccess); | ||||||
|  | @ -131,6 +224,18 @@ namespace ServiceLib.ViewModels | ||||||
| 
 | 
 | ||||||
|         #endregion Refresh Save |         #endregion Refresh Save | ||||||
| 
 | 
 | ||||||
|  |         private async Task RoutingBasicImportRules() | ||||||
|  |         { | ||||||
|  |             //Extra to bypass the mainland | ||||||
|  |             ProxyDomain = "geosite:google"; | ||||||
|  |             DirectDomain = "geosite:cn"; | ||||||
|  |             DirectIP = "geoip:private,geoip:cn"; | ||||||
|  |             BlockDomain = "geosite:category-ads-all"; | ||||||
|  | 
 | ||||||
|  |             //NoticeHandler.Instance.Enqueue(ResUI.OperationSuccess); | ||||||
|  |             NoticeHandler.Instance.Enqueue(ResUI.OperationSuccess); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         public async Task RoutingAdvancedEditAsync(bool blNew) |         public async Task RoutingAdvancedEditAsync(bool blNew) | ||||||
|         { |         { | ||||||
|             RoutingItem item; |             RoutingItem item; | ||||||
|  |  | ||||||
|  | @ -347,6 +347,11 @@ namespace ServiceLib.ViewModels | ||||||
|         public async Task RefreshRoutingsMenu() |         public async Task RefreshRoutingsMenu() | ||||||
|         { |         { | ||||||
|             _routingItems.Clear(); |             _routingItems.Clear(); | ||||||
|  |             if (!_config.RoutingBasicItem.EnableRoutingAdvanced) | ||||||
|  |             { | ||||||
|  |                 BlRouting = false; | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
| 
 | 
 | ||||||
|             BlRouting = true; |             BlRouting = true; | ||||||
|             var routings = await AppHandler.Instance.RoutingItems(); |             var routings = await AppHandler.Instance.RoutingItems(); | ||||||
|  | @ -469,7 +474,7 @@ namespace ServiceLib.ViewModels | ||||||
|             } |             } | ||||||
|             else |             else | ||||||
|             { |             { | ||||||
|                 InboundLanDisplay = $"{ResUI.LabLAN}:{Global.None}"; |                 InboundLanDisplay = $"{ResUI.LabLAN}:None"; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -15,6 +15,7 @@ | ||||||
|     mc:Ignorable="d"> |     mc:Ignorable="d"> | ||||||
| 
 | 
 | ||||||
|     <DockPanel> |     <DockPanel> | ||||||
|  | 
 | ||||||
|         <StackPanel |         <StackPanel | ||||||
|             Classes="Margin8" |             Classes="Margin8" | ||||||
|             DockPanel.Dock="Top" |             DockPanel.Dock="Top" | ||||||
|  | @ -54,6 +55,7 @@ | ||||||
|                 Margin="8,0,0,0" /> |                 Margin="8,0,0,0" /> | ||||||
|         </StackPanel> |         </StackPanel> | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|         <StackPanel |         <StackPanel | ||||||
|             HorizontalAlignment="Right" |             HorizontalAlignment="Right" | ||||||
|             Classes="Margin8" |             Classes="Margin8" | ||||||
|  | @ -132,6 +134,9 @@ | ||||||
|                     </DataGrid> |                     </DataGrid> | ||||||
|                 </TabItem> |                 </TabItem> | ||||||
|             </TabControl> |             </TabControl> | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|         </DockPanel> |         </DockPanel> | ||||||
|  | 
 | ||||||
|     </DockPanel> |     </DockPanel> | ||||||
| </Window> | </Window> | ||||||
|  | @ -54,6 +54,11 @@ namespace v2rayN.Handler | ||||||
|         { |         { | ||||||
|             try |             try | ||||||
|             { |             { | ||||||
|  |                 if (!config.RoutingBasicItem.EnableRoutingAdvanced) | ||||||
|  |                 { | ||||||
|  |                     return null; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|                 var item = await ConfigHandler.GetDefaultRouting(config); |                 var item = await ConfigHandler.GetDefaultRouting(config); | ||||||
|                 if (item == null || Utils.IsNullOrEmpty(item.CustomIcon) || !File.Exists(item.CustomIcon)) |                 if (item == null || Utils.IsNullOrEmpty(item.CustomIcon) || !File.Exists(item.CustomIcon)) | ||||||
|                 { |                 { | ||||||
|  |  | ||||||
|  | @ -28,6 +28,23 @@ | ||||||
|                 VerticalAlignment="Center" |                 VerticalAlignment="Center" | ||||||
|                 ClipToBounds="True" |                 ClipToBounds="True" | ||||||
|                 Style="{StaticResource MaterialDesignToolBar}"> |                 Style="{StaticResource MaterialDesignToolBar}"> | ||||||
|  |                 <Menu Margin="0,1" Style="{StaticResource ToolbarMenu}"> | ||||||
|  |                     <MenuItem x:Name="menuRoutingBasic" Padding="8,0"> | ||||||
|  |                         <MenuItem.Header> | ||||||
|  |                             <StackPanel Orientation="Horizontal"> | ||||||
|  |                                 <materialDesign:PackIcon | ||||||
|  |                                     Margin="{StaticResource MarginRight8}" | ||||||
|  |                                     VerticalAlignment="Center" | ||||||
|  |                                     Kind="Server" /> | ||||||
|  |                                 <TextBlock Text="{x:Static resx:ResUI.menuRoutingBasic}" /> | ||||||
|  |                             </StackPanel> | ||||||
|  |                         </MenuItem.Header> | ||||||
|  |                         <MenuItem | ||||||
|  |                             x:Name="menuRoutingBasicImportRules" | ||||||
|  |                             Height="{StaticResource MenuItemHeight}" | ||||||
|  |                             Header="{x:Static resx:ResUI.menuRoutingBasicImportRules}" /> | ||||||
|  |                     </MenuItem> | ||||||
|  |                 </Menu> | ||||||
|                 <Menu Margin="0,1" Style="{StaticResource ToolbarMenu}"> |                 <Menu Margin="0,1" Style="{StaticResource ToolbarMenu}"> | ||||||
|                     <MenuItem x:Name="menuRoutingAdvanced" Padding="8,0"> |                     <MenuItem x:Name="menuRoutingAdvanced" Padding="8,0"> | ||||||
|                         <MenuItem.Header> |                         <MenuItem.Header> | ||||||
|  | @ -49,6 +66,16 @@ | ||||||
|                             Header="{x:Static resx:ResUI.menuRoutingAdvancedImportRules}" /> |                             Header="{x:Static resx:ResUI.menuRoutingAdvancedImportRules}" /> | ||||||
|                     </MenuItem> |                     </MenuItem> | ||||||
|                 </Menu> |                 </Menu> | ||||||
|  |                 <!--<Separator /> | ||||||
|  |                 <TextBlock | ||||||
|  |                     Margin="{StaticResource MarginLeft8}" | ||||||
|  |                     VerticalAlignment="Center" | ||||||
|  |                     Style="{StaticResource ToolbarTextBlock}" | ||||||
|  |                     Text="{x:Static resx:ResUI.TbenableRoutingAdvanced}" /> | ||||||
|  |                 <ToggleButton | ||||||
|  |                     x:Name="togenableRoutingAdvanced" | ||||||
|  |                     Margin="{StaticResource MarginLeft8}" | ||||||
|  |                     HorizontalAlignment="Left" />--> | ||||||
|                 <Separator /> |                 <Separator /> | ||||||
|                 <TextBlock |                 <TextBlock | ||||||
|                     Margin="{StaticResource MarginLeft8}" |                     Margin="{StaticResource MarginLeft8}" | ||||||
|  | @ -197,6 +224,107 @@ | ||||||
|                     </DataGrid> |                     </DataGrid> | ||||||
|                 </TabItem> |                 </TabItem> | ||||||
|             </TabControl> |             </TabControl> | ||||||
|  | 
 | ||||||
|  |             <TabControl x:Name="tabBasic"> | ||||||
|  |                 <TabItem Header="{x:Static resx:ResUI.TbRoutingTabProxy}"> | ||||||
|  |                     <Grid Margin="{StaticResource Margin8}"> | ||||||
|  |                         <Grid.ColumnDefinitions> | ||||||
|  |                             <ColumnDefinition Width="1*" /> | ||||||
|  |                             <ColumnDefinition Width="10" /> | ||||||
|  |                             <ColumnDefinition Width="1*" /> | ||||||
|  |                         </Grid.ColumnDefinitions> | ||||||
|  |                         <GroupBox | ||||||
|  |                             Grid.Column="0" | ||||||
|  |                             Header="Domain" | ||||||
|  |                             Style="{StaticResource MyGroupBox}"> | ||||||
|  |                             <TextBox | ||||||
|  |                                 Name="txtProxyDomain" | ||||||
|  |                                 AcceptsReturn="True" | ||||||
|  |                                 Style="{StaticResource DefTextBox}" | ||||||
|  |                                 TextWrapping="Wrap" | ||||||
|  |                                 VerticalScrollBarVisibility="Auto" /> | ||||||
|  |                         </GroupBox> | ||||||
|  |                         <GridSplitter Grid.Column="1" HorizontalAlignment="Stretch" /> | ||||||
|  |                         <GroupBox | ||||||
|  |                             Grid.Column="2" | ||||||
|  |                             Header="IP" | ||||||
|  |                             Style="{StaticResource MyGroupBox}"> | ||||||
|  |                             <TextBox | ||||||
|  |                                 Name="txtProxyIP" | ||||||
|  |                                 AcceptsReturn="True" | ||||||
|  |                                 Style="{StaticResource DefTextBox}" | ||||||
|  |                                 TextWrapping="Wrap" | ||||||
|  |                                 VerticalScrollBarVisibility="Auto" /> | ||||||
|  |                         </GroupBox> | ||||||
|  |                     </Grid> | ||||||
|  |                 </TabItem> | ||||||
|  | 
 | ||||||
|  |                 <TabItem Header="{x:Static resx:ResUI.TbRoutingTabDirect}"> | ||||||
|  |                     <Grid Margin="{StaticResource Margin8}"> | ||||||
|  |                         <Grid.ColumnDefinitions> | ||||||
|  |                             <ColumnDefinition Width="1*" /> | ||||||
|  |                             <ColumnDefinition Width="10" /> | ||||||
|  |                             <ColumnDefinition Width="1*" /> | ||||||
|  |                         </Grid.ColumnDefinitions> | ||||||
|  |                         <GroupBox | ||||||
|  |                             Grid.Column="0" | ||||||
|  |                             Header="Domain" | ||||||
|  |                             Style="{StaticResource MyGroupBox}"> | ||||||
|  |                             <TextBox | ||||||
|  |                                 Name="txtDirectDomain" | ||||||
|  |                                 AcceptsReturn="True" | ||||||
|  |                                 Style="{StaticResource DefTextBox}" | ||||||
|  |                                 TextWrapping="Wrap" | ||||||
|  |                                 VerticalScrollBarVisibility="Auto" /> | ||||||
|  |                         </GroupBox> | ||||||
|  |                         <GridSplitter Grid.Column="1" HorizontalAlignment="Stretch" /> | ||||||
|  |                         <GroupBox | ||||||
|  |                             Grid.Column="2" | ||||||
|  |                             Header="IP" | ||||||
|  |                             Style="{StaticResource MyGroupBox}"> | ||||||
|  |                             <TextBox | ||||||
|  |                                 Name="txtDirectIP" | ||||||
|  |                                 AcceptsReturn="True" | ||||||
|  |                                 Style="{StaticResource DefTextBox}" | ||||||
|  |                                 TextWrapping="Wrap" | ||||||
|  |                                 VerticalScrollBarVisibility="Auto" /> | ||||||
|  |                         </GroupBox> | ||||||
|  |                     </Grid> | ||||||
|  |                 </TabItem> | ||||||
|  | 
 | ||||||
|  |                 <TabItem Header="{x:Static resx:ResUI.TbRoutingTabBlock}"> | ||||||
|  |                     <Grid Margin="{StaticResource Margin8}"> | ||||||
|  |                         <Grid.ColumnDefinitions> | ||||||
|  |                             <ColumnDefinition Width="1*" /> | ||||||
|  |                             <ColumnDefinition Width="10" /> | ||||||
|  |                             <ColumnDefinition Width="1*" /> | ||||||
|  |                         </Grid.ColumnDefinitions> | ||||||
|  |                         <GroupBox | ||||||
|  |                             Grid.Column="0" | ||||||
|  |                             Header="Domain" | ||||||
|  |                             Style="{StaticResource MyGroupBox}"> | ||||||
|  |                             <TextBox | ||||||
|  |                                 Name="txtBlockDomain" | ||||||
|  |                                 AcceptsReturn="True" | ||||||
|  |                                 Style="{StaticResource DefTextBox}" | ||||||
|  |                                 TextWrapping="Wrap" | ||||||
|  |                                 VerticalScrollBarVisibility="Auto" /> | ||||||
|  |                         </GroupBox> | ||||||
|  |                         <GridSplitter Grid.Column="1" HorizontalAlignment="Stretch" /> | ||||||
|  |                         <GroupBox | ||||||
|  |                             Grid.Column="2" | ||||||
|  |                             Header="IP" | ||||||
|  |                             Style="{StaticResource MyGroupBox}"> | ||||||
|  |                             <TextBox | ||||||
|  |                                 Name="txtBlockIP" | ||||||
|  |                                 AcceptsReturn="True" | ||||||
|  |                                 Style="{StaticResource DefTextBox}" | ||||||
|  |                                 TextWrapping="Wrap" | ||||||
|  |                                 VerticalScrollBarVisibility="Auto" /> | ||||||
|  |                         </GroupBox> | ||||||
|  |                     </Grid> | ||||||
|  |                 </TabItem> | ||||||
|  |             </TabControl> | ||||||
|         </DockPanel> |         </DockPanel> | ||||||
|     </DockPanel> |     </DockPanel> | ||||||
| </reactiveui:ReactiveWindow> | </reactiveui:ReactiveWindow> | ||||||
|  | @ -39,10 +39,24 @@ namespace v2rayN.Views | ||||||
|                 this.OneWayBind(ViewModel, vm => vm.RoutingItems, v => v.lstRoutings.ItemsSource).DisposeWith(disposables); |                 this.OneWayBind(ViewModel, vm => vm.RoutingItems, v => v.lstRoutings.ItemsSource).DisposeWith(disposables); | ||||||
|                 this.Bind(ViewModel, vm => vm.SelectedSource, v => v.lstRoutings.SelectedItem).DisposeWith(disposables); |                 this.Bind(ViewModel, vm => vm.SelectedSource, v => v.lstRoutings.SelectedItem).DisposeWith(disposables); | ||||||
| 
 | 
 | ||||||
|  |                 //this.Bind(ViewModel, vm => vm.enableRoutingAdvanced, v => v.togenableRoutingAdvanced.IsChecked).DisposeWith(disposables); | ||||||
|                 this.Bind(ViewModel, vm => vm.domainStrategy, v => v.cmbdomainStrategy.Text).DisposeWith(disposables); |                 this.Bind(ViewModel, vm => vm.domainStrategy, v => v.cmbdomainStrategy.Text).DisposeWith(disposables); | ||||||
|                 this.Bind(ViewModel, vm => vm.domainMatcher, v => v.cmbdomainMatcher.Text).DisposeWith(disposables); |                 this.Bind(ViewModel, vm => vm.domainMatcher, v => v.cmbdomainMatcher.Text).DisposeWith(disposables); | ||||||
|                 this.Bind(ViewModel, vm => vm.domainStrategy4Singbox, v => v.cmbdomainStrategy4Singbox.Text).DisposeWith(disposables); |                 this.Bind(ViewModel, vm => vm.domainStrategy4Singbox, v => v.cmbdomainStrategy4Singbox.Text).DisposeWith(disposables); | ||||||
| 
 | 
 | ||||||
|  |                 this.Bind(ViewModel, vm => vm.ProxyDomain, v => v.txtProxyDomain.Text).DisposeWith(disposables); | ||||||
|  |                 this.Bind(ViewModel, vm => vm.ProxyIP, v => v.txtProxyIP.Text).DisposeWith(disposables); | ||||||
|  |                 this.Bind(ViewModel, vm => vm.DirectDomain, v => v.txtDirectDomain.Text).DisposeWith(disposables); | ||||||
|  |                 this.Bind(ViewModel, vm => vm.DirectIP, v => v.txtDirectIP.Text).DisposeWith(disposables); | ||||||
|  |                 this.Bind(ViewModel, vm => vm.BlockDomain, v => v.txtBlockDomain.Text).DisposeWith(disposables); | ||||||
|  |                 this.Bind(ViewModel, vm => vm.BlockIP, v => v.txtBlockIP.Text).DisposeWith(disposables); | ||||||
|  | 
 | ||||||
|  |                 this.OneWayBind(ViewModel, vm => vm.enableRoutingBasic, v => v.menuRoutingBasic.Visibility).DisposeWith(disposables); | ||||||
|  |                 this.OneWayBind(ViewModel, vm => vm.enableRoutingAdvanced, v => v.menuRoutingAdvanced.Visibility).DisposeWith(disposables); | ||||||
|  |                 this.OneWayBind(ViewModel, vm => vm.enableRoutingBasic, v => v.tabBasic.Visibility).DisposeWith(disposables); | ||||||
|  |                 this.OneWayBind(ViewModel, vm => vm.enableRoutingAdvanced, v => v.tabAdvanced.Visibility).DisposeWith(disposables); | ||||||
|  | 
 | ||||||
|  |                 this.BindCommand(ViewModel, vm => vm.RoutingBasicImportRulesCmd, v => v.menuRoutingBasicImportRules).DisposeWith(disposables); | ||||||
|                 this.BindCommand(ViewModel, vm => vm.RoutingAdvancedAddCmd, v => v.menuRoutingAdvancedAdd).DisposeWith(disposables); |                 this.BindCommand(ViewModel, vm => vm.RoutingAdvancedAddCmd, v => v.menuRoutingAdvancedAdd).DisposeWith(disposables); | ||||||
|                 this.BindCommand(ViewModel, vm => vm.RoutingAdvancedAddCmd, v => v.menuRoutingAdvancedAdd2).DisposeWith(disposables); |                 this.BindCommand(ViewModel, vm => vm.RoutingAdvancedAddCmd, v => v.menuRoutingAdvancedAdd2).DisposeWith(disposables); | ||||||
|                 this.BindCommand(ViewModel, vm => vm.RoutingAdvancedRemoveCmd, v => v.menuRoutingAdvancedRemove).DisposeWith(disposables); |                 this.BindCommand(ViewModel, vm => vm.RoutingAdvancedRemoveCmd, v => v.menuRoutingAdvancedRemove).DisposeWith(disposables); | ||||||
|  | @ -88,6 +102,11 @@ namespace v2rayN.Views | ||||||
| 
 | 
 | ||||||
|         private void RoutingSettingWindow_PreviewKeyDown(object sender, KeyEventArgs e) |         private void RoutingSettingWindow_PreviewKeyDown(object sender, KeyEventArgs e) | ||||||
|         { |         { | ||||||
|  |             if (ViewModel?.enableRoutingBasic ?? false) | ||||||
|  |             { | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|             if (Keyboard.IsKeyDown(Key.LeftCtrl) || Keyboard.IsKeyDown(Key.RightCtrl)) |             if (Keyboard.IsKeyDown(Key.LeftCtrl) || Keyboard.IsKeyDown(Key.RightCtrl)) | ||||||
|             { |             { | ||||||
|                 if (e.Key == Key.A) |                 if (e.Key == Key.A) | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue