mirror of
				https://github.com/2dust/v2rayN.git
				synced 2025-10-26 18:24:43 +00:00 
			
		
		
		
	Refactor the Linux version to not store sudo password
This commit is contained in:
		
							parent
							
								
									ea42246d1b
								
							
						
					
					
						commit
						c6d347d49a
					
				
					 18 changed files with 161 additions and 77 deletions
				
			
		|  | @ -20,6 +20,7 @@ public enum EViewAction | ||||||
|     BrowseServer, |     BrowseServer, | ||||||
|     ImportRulesFromFile, |     ImportRulesFromFile, | ||||||
|     InitSettingFont, |     InitSettingFont, | ||||||
|  |     PasswordInput, | ||||||
|     SubEditWindow, |     SubEditWindow, | ||||||
|     RoutingRuleSettingWindow, |     RoutingRuleSettingWindow, | ||||||
|     RoutingRuleDetailsWindow, |     RoutingRuleDetailsWindow, | ||||||
|  |  | ||||||
|  | @ -40,6 +40,8 @@ public sealed class AppHandler | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     public string LinuxSudoPwd { get; set; } | ||||||
|  | 
 | ||||||
|     #endregion Property |     #endregion Property | ||||||
| 
 | 
 | ||||||
|     #region Init |     #region Init | ||||||
|  |  | ||||||
|  | @ -229,9 +229,7 @@ public class CoreHandler | ||||||
|     { |     { | ||||||
|         return _config.TunModeItem.EnableTun |         return _config.TunModeItem.EnableTun | ||||||
|                && eCoreType == ECoreType.sing_box |                && eCoreType == ECoreType.sing_box | ||||||
|                && (Utils.IsNonWindows()) |                && Utils.IsNonWindows(); | ||||||
|             //&& _config.TunModeItem.LinuxSudoPwd.IsNotEmpty() |  | ||||||
|             ; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #endregion Private |     #endregion Private | ||||||
|  | @ -288,13 +286,11 @@ public class CoreHandler | ||||||
|             } |             } | ||||||
|             proc.Start(); |             proc.Start(); | ||||||
| 
 | 
 | ||||||
|             if (isNeedSudo && _config.TunModeItem.LinuxSudoPwd.IsNotEmpty()) |             if (isNeedSudo && AppHandler.Instance.LinuxSudoPwd.IsNotEmpty()) | ||||||
|             { |             { | ||||||
|                 var pwd = DesUtils.Decrypt(_config.TunModeItem.LinuxSudoPwd); |                 await proc.StandardInput.WriteLineAsync(); | ||||||
|                 await Task.Delay(10); |                 await Task.Delay(10); | ||||||
|                 await proc.StandardInput.WriteLineAsync(pwd); |                 await proc.StandardInput.WriteLineAsync(AppHandler.Instance.LinuxSudoPwd); | ||||||
|                 await Task.Delay(10); |  | ||||||
|                 await proc.StandardInput.WriteLineAsync(pwd); |  | ||||||
|             } |             } | ||||||
|             if (isNeedSudo) |             if (isNeedSudo) | ||||||
|                 _linuxSudoPid = proc.Id; |                 _linuxSudoPid = proc.Id; | ||||||
|  | @ -333,7 +329,7 @@ public class CoreHandler | ||||||
|         proc.StartInfo.FileName = shFilePath; |         proc.StartInfo.FileName = shFilePath; | ||||||
|         proc.StartInfo.Arguments = ""; |         proc.StartInfo.Arguments = ""; | ||||||
|         proc.StartInfo.WorkingDirectory = ""; |         proc.StartInfo.WorkingDirectory = ""; | ||||||
|         if (_config.TunModeItem.LinuxSudoPwd.IsNotEmpty()) |         if (AppHandler.Instance.LinuxSudoPwd.IsNotEmpty()) | ||||||
|         { |         { | ||||||
|             proc.StartInfo.StandardInputEncoding = Encoding.UTF8; |             proc.StartInfo.StandardInputEncoding = Encoding.UTF8; | ||||||
|             proc.StartInfo.RedirectStandardInput = true; |             proc.StartInfo.RedirectStandardInput = true; | ||||||
|  | @ -342,7 +338,7 @@ public class CoreHandler | ||||||
| 
 | 
 | ||||||
|     private async Task KillProcessAsLinuxSudo() |     private async Task KillProcessAsLinuxSudo() | ||||||
|     { |     { | ||||||
|         var cmdLine = $"kill {_linuxSudoPid}"; |         var cmdLine = $"pkill -P {_linuxSudoPid} ; kill {_linuxSudoPid}"; | ||||||
|         var shFilePath = await CreateLinuxShellFile(cmdLine, "kill_as_sudo.sh"); |         var shFilePath = await CreateLinuxShellFile(cmdLine, "kill_as_sudo.sh"); | ||||||
|         Process proc = new() |         Process proc = new() | ||||||
|         { |         { | ||||||
|  | @ -357,15 +353,13 @@ public class CoreHandler | ||||||
|         }; |         }; | ||||||
|         proc.Start(); |         proc.Start(); | ||||||
| 
 | 
 | ||||||
|         if (_config.TunModeItem.LinuxSudoPwd.IsNotEmpty()) |         if (AppHandler.Instance.LinuxSudoPwd.IsNotEmpty()) | ||||||
|         { |         { | ||||||
|             try |             try | ||||||
|             { |             { | ||||||
|                 var pwd = DesUtils.Decrypt(_config.TunModeItem.LinuxSudoPwd); |                 await proc.StandardInput.WriteLineAsync(); | ||||||
|                 await Task.Delay(10); |                 await Task.Delay(10); | ||||||
|                 await proc.StandardInput.WriteLineAsync(pwd); |                 await proc.StandardInput.WriteLineAsync(AppHandler.Instance.LinuxSudoPwd); | ||||||
|                 await Task.Delay(10); |  | ||||||
|                 await proc.StandardInput.WriteLineAsync(pwd); |  | ||||||
|             } |             } | ||||||
|             catch (Exception) |             catch (Exception) | ||||||
|             { |             { | ||||||
|  | @ -375,7 +369,7 @@ public class CoreHandler | ||||||
| 
 | 
 | ||||||
|         var timeout = new CancellationTokenSource(TimeSpan.FromSeconds(10)); |         var timeout = new CancellationTokenSource(TimeSpan.FromSeconds(10)); | ||||||
|         await proc.WaitForExitAsync(timeout.Token); |         await proc.WaitForExitAsync(timeout.Token); | ||||||
|         await Task.Delay(3000); |         await Task.Delay(1000); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private async Task<string> CreateLinuxShellFile(string cmdLine, string fileName) |     private async Task<string> CreateLinuxShellFile(string cmdLine, string fileName) | ||||||
|  | @ -389,10 +383,6 @@ public class CoreHandler | ||||||
|         { |         { | ||||||
|             sb.AppendLine($"{cmdLine}"); |             sb.AppendLine($"{cmdLine}"); | ||||||
|         } |         } | ||||||
|         else if (_config.TunModeItem.LinuxSudoPwd.IsNullOrEmpty()) |  | ||||||
|         { |  | ||||||
|             sb.AppendLine($"pkexec {cmdLine}"); |  | ||||||
|         } |  | ||||||
|         else |         else | ||||||
|         { |         { | ||||||
|             sb.AppendLine($"sudo -S {cmdLine}"); |             sb.AppendLine($"sudo -S {cmdLine}"); | ||||||
|  |  | ||||||
|  | @ -147,7 +147,6 @@ public class TunModeItem | ||||||
|     public int Mtu { get; set; } |     public int Mtu { get; set; } | ||||||
|     public bool EnableExInbound { get; set; } |     public bool EnableExInbound { get; set; } | ||||||
|     public bool EnableIPv6Address { get; set; } |     public bool EnableIPv6Address { get; set; } | ||||||
|     public string? LinuxSudoPwd { get; set; } |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| [Serializable] | [Serializable] | ||||||
|  |  | ||||||
							
								
								
									
										2
									
								
								v2rayN/ServiceLib/Resx/ResUI.Designer.cs
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2
									
								
								v2rayN/ServiceLib/Resx/ResUI.Designer.cs
									
									
									
										generated
									
									
									
								
							|  | @ -3247,7 +3247,7 @@ namespace ServiceLib.Resx { | ||||||
|         } |         } | ||||||
|          |          | ||||||
|         /// <summary> |         /// <summary> | ||||||
|         ///   查找类似 The password is encrypted and stored only in local files 的本地化字符串。 |         ///   查找类似 The password you entered cannot be verified, so make sure you enter it correctly. If the application does not work properly due to an incorrect input, please restart the application. The password will not be stored and you will need to enter it again after each restart. 的本地化字符串。 | ||||||
|         /// </summary> |         /// </summary> | ||||||
|         public static string TbSettingsLinuxSudoPasswordTip { |         public static string TbSettingsLinuxSudoPasswordTip { | ||||||
|             get { |             get { | ||||||
|  |  | ||||||
|  | @ -1339,7 +1339,7 @@ | ||||||
|     <value>رمز عبور sudo سیستم</value> |     <value>رمز عبور sudo سیستم</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="TbSettingsLinuxSudoPasswordTip" xml:space="preserve"> |   <data name="TbSettingsLinuxSudoPasswordTip" xml:space="preserve"> | ||||||
|     <value>رمز عبور رمزگذاری شده و فقط در فایل های محلی ذخیره می شود.</value> |     <value>The password you entered cannot be verified, so make sure you enter it correctly. If the application does not work properly due to an incorrect input, please restart the application. The password will not be stored and you will need to enter it again after each restart.</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="TbSettingsLinuxSudoPasswordIsEmpty" xml:space="preserve"> |   <data name="TbSettingsLinuxSudoPasswordIsEmpty" xml:space="preserve"> | ||||||
|     <value>لطفاً ابتدا رمز عبور sudo را در تنظیمات حالت Tun تنظیم کنید</value> |     <value>لطفاً ابتدا رمز عبور sudo را در تنظیمات حالت Tun تنظیم کنید</value> | ||||||
|  | @ -1416,4 +1416,4 @@ | ||||||
|   <data name="menuExportConfig" xml:space="preserve"> |   <data name="menuExportConfig" xml:space="preserve"> | ||||||
|     <value>صادر کردن سرور</value> |     <value>صادر کردن سرور</value> | ||||||
|   </data> |   </data> | ||||||
| </root> | </root> | ||||||
|  | @ -1339,7 +1339,7 @@ | ||||||
|     <value>Rendszer sudo jelszó</value> |     <value>Rendszer sudo jelszó</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="TbSettingsLinuxSudoPasswordTip" xml:space="preserve"> |   <data name="TbSettingsLinuxSudoPasswordTip" xml:space="preserve"> | ||||||
|     <value>A jelszó titkosítva és csak a helyi fájlokban tárolva.</value> |     <value>The password you entered cannot be verified, so make sure you enter it correctly. If the application does not work properly due to an incorrect input, please restart the application. The password will not be stored and you will need to enter it again after each restart.</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="TbSettingsLinuxSudoPasswordIsEmpty" xml:space="preserve"> |   <data name="TbSettingsLinuxSudoPasswordIsEmpty" xml:space="preserve"> | ||||||
|     <value>Kérlek, először állítsd be a sudo jelszót a Tun módban</value> |     <value>Kérlek, először állítsd be a sudo jelszót a Tun módban</value> | ||||||
|  |  | ||||||
|  | @ -1339,7 +1339,7 @@ | ||||||
|     <value>System sudo password</value> |     <value>System sudo password</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="TbSettingsLinuxSudoPasswordTip" xml:space="preserve"> |   <data name="TbSettingsLinuxSudoPasswordTip" xml:space="preserve"> | ||||||
|     <value>The password is encrypted and stored only in local files</value> |     <value>The password you entered cannot be verified, so make sure you enter it correctly. If the application does not work properly due to an incorrect input, please restart the application. The password will not be stored and you will need to enter it again after each restart.</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="TbSettingsLinuxSudoPasswordIsEmpty" xml:space="preserve"> |   <data name="TbSettingsLinuxSudoPasswordIsEmpty" xml:space="preserve"> | ||||||
|     <value>Please set the sudo password in Tun mode settings first</value> |     <value>Please set the sudo password in Tun mode settings first</value> | ||||||
|  |  | ||||||
|  | @ -1339,7 +1339,7 @@ | ||||||
|     <value>System sudo password</value> |     <value>System sudo password</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="TbSettingsLinuxSudoPasswordTip" xml:space="preserve"> |   <data name="TbSettingsLinuxSudoPasswordTip" xml:space="preserve"> | ||||||
|     <value>Пароль зашифрован и хранится только в локальных файлах..</value> |     <value>The password you entered cannot be verified, so make sure you enter it correctly. If the application does not work properly due to an incorrect input, please restart the application. The password will not be stored and you will need to enter it again after each restart.</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="TbSettingsLinuxSudoPasswordIsEmpty" xml:space="preserve"> |   <data name="TbSettingsLinuxSudoPasswordIsEmpty" xml:space="preserve"> | ||||||
|     <value>Please set the sudo password in Tun mode settings first</value> |     <value>Please set the sudo password in Tun mode settings first</value> | ||||||
|  |  | ||||||
|  | @ -1071,13 +1071,13 @@ | ||||||
|   <data name="TbHeaderType8" xml:space="preserve"> |   <data name="TbHeaderType8" xml:space="preserve"> | ||||||
|     <value>拥塞控制算法</value> |     <value>拥塞控制算法</value> | ||||||
|   </data> |   </data> | ||||||
|     <data name="LvPrevProfile" xml:space="preserve"> |   <data name="LvPrevProfile" xml:space="preserve"> | ||||||
|     <value>前置代理配置文件别名</value> |     <value>前置代理配置文件别名</value> | ||||||
|   </data> |   </data> | ||||||
|     <data name="LvNextProfile" xml:space="preserve"> |   <data name="LvNextProfile" xml:space="preserve"> | ||||||
|     <value>落地代理配置文件別名</value> |     <value>落地代理配置文件別名</value> | ||||||
|   </data> |   </data> | ||||||
|     <data name="LvPrevProfileTip" xml:space="preserve"> |   <data name="LvPrevProfileTip" xml:space="preserve"> | ||||||
|     <value>请确保配置文件别名存在并唯一</value> |     <value>请确保配置文件别名存在并唯一</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="TbSettingsEnableExInbound" xml:space="preserve"> |   <data name="TbSettingsEnableExInbound" xml:space="preserve"> | ||||||
|  | @ -1336,7 +1336,7 @@ | ||||||
|     <value>系统的 sudo 密码</value> |     <value>系统的 sudo 密码</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="TbSettingsLinuxSudoPasswordTip" xml:space="preserve"> |   <data name="TbSettingsLinuxSudoPasswordTip" xml:space="preserve"> | ||||||
|     <value>密码已加密且只存储在本地文件中,无密码则每次都要输入</value> |     <value>输入的密码无法校验,所以请确保输入正确。如果因为输入错误导致无法正常运行时,请重启本应用。 密码不会存储,每次重启后都需要再次输入。</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="TbSettingsLinuxSudoPasswordIsEmpty" xml:space="preserve"> |   <data name="TbSettingsLinuxSudoPasswordIsEmpty" xml:space="preserve"> | ||||||
|     <value>请先在 Tun 模式设置中设置 sudo 密码</value> |     <value>请先在 Tun 模式设置中设置 sudo 密码</value> | ||||||
|  | @ -1413,4 +1413,4 @@ | ||||||
|   <data name="menuExportConfig" xml:space="preserve"> |   <data name="menuExportConfig" xml:space="preserve"> | ||||||
|     <value>导出配置文件</value> |     <value>导出配置文件</value> | ||||||
|   </data> |   </data> | ||||||
| </root> | </root> | ||||||
|  | @ -1071,13 +1071,13 @@ | ||||||
|   <data name="TbHeaderType8" xml:space="preserve"> |   <data name="TbHeaderType8" xml:space="preserve"> | ||||||
|     <value>擁塞控制算法</value> |     <value>擁塞控制算法</value> | ||||||
|   </data> |   </data> | ||||||
|     <data name="LvPrevProfile" xml:space="preserve"> |   <data name="LvPrevProfile" xml:space="preserve"> | ||||||
|     <value>前置代理設定檔別名</value> |     <value>前置代理設定檔別名</value> | ||||||
|   </data> |   </data> | ||||||
|     <data name="LvNextProfile" xml:space="preserve"> |   <data name="LvNextProfile" xml:space="preserve"> | ||||||
|     <value>落地代理設定檔別名</value> |     <value>落地代理設定檔別名</value> | ||||||
|   </data> |   </data> | ||||||
|     <data name="LvPrevProfileTip" xml:space="preserve"> |   <data name="LvPrevProfileTip" xml:space="preserve"> | ||||||
|     <value>請確保設定檔別名存在並且唯一</value> |     <value>請確保設定檔別名存在並且唯一</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="TbSettingsEnableExInbound" xml:space="preserve"> |   <data name="TbSettingsEnableExInbound" xml:space="preserve"> | ||||||
|  | @ -1336,7 +1336,7 @@ | ||||||
|     <value>系統的 sudo 密碼</value> |     <value>系統的 sudo 密碼</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="TbSettingsLinuxSudoPasswordTip" xml:space="preserve"> |   <data name="TbSettingsLinuxSudoPasswordTip" xml:space="preserve"> | ||||||
|     <value>密碼已加密且只儲存在本機檔案中,無密碼則每次都要輸入</value> |     <value>輸入的密碼無法校驗,所以請確保輸入正確。如果因為輸入錯誤導致無法正常運作時,請重新啟動本應用。 密碼不會存儲,每次重啟後都需要再次輸入。</value> | ||||||
|   </data> |   </data> | ||||||
|   <data name="TbSettingsLinuxSudoPasswordIsEmpty" xml:space="preserve"> |   <data name="TbSettingsLinuxSudoPasswordIsEmpty" xml:space="preserve"> | ||||||
|     <value>請先在 Tun 模式設定中設定 sudo 密碼</value> |     <value>請先在 Tun 模式設定中設定 sudo 密碼</value> | ||||||
|  | @ -1413,4 +1413,4 @@ | ||||||
|   <data name="menuExportConfig" xml:space="preserve"> |   <data name="menuExportConfig" xml:space="preserve"> | ||||||
|     <value>匯出設定檔</value> |     <value>匯出設定檔</value> | ||||||
|   </data> |   </data> | ||||||
| </root> | </root> | ||||||
|  | @ -88,7 +88,6 @@ public class OptionSettingViewModel : MyReactiveObject | ||||||
|     [Reactive] public int TunMtu { get; set; } |     [Reactive] public int TunMtu { get; set; } | ||||||
|     [Reactive] public bool TunEnableExInbound { get; set; } |     [Reactive] public bool TunEnableExInbound { get; set; } | ||||||
|     [Reactive] public bool TunEnableIPv6Address { get; set; } |     [Reactive] public bool TunEnableIPv6Address { get; set; } | ||||||
|     [Reactive] public string TunLinuxSudoPassword { get; set; } |  | ||||||
| 
 | 
 | ||||||
|     #endregion Tun mode |     #endregion Tun mode | ||||||
| 
 | 
 | ||||||
|  | @ -205,7 +204,6 @@ public class OptionSettingViewModel : MyReactiveObject | ||||||
|         TunMtu = _config.TunModeItem.Mtu; |         TunMtu = _config.TunModeItem.Mtu; | ||||||
|         TunEnableExInbound = _config.TunModeItem.EnableExInbound; |         TunEnableExInbound = _config.TunModeItem.EnableExInbound; | ||||||
|         TunEnableIPv6Address = _config.TunModeItem.EnableIPv6Address; |         TunEnableIPv6Address = _config.TunModeItem.EnableIPv6Address; | ||||||
|         TunLinuxSudoPassword = _config.TunModeItem.LinuxSudoPwd; |  | ||||||
| 
 | 
 | ||||||
|         #endregion Tun mode |         #endregion Tun mode | ||||||
| 
 | 
 | ||||||
|  | @ -358,11 +356,7 @@ public class OptionSettingViewModel : MyReactiveObject | ||||||
|         _config.TunModeItem.Mtu = TunMtu; |         _config.TunModeItem.Mtu = TunMtu; | ||||||
|         _config.TunModeItem.EnableExInbound = TunEnableExInbound; |         _config.TunModeItem.EnableExInbound = TunEnableExInbound; | ||||||
|         _config.TunModeItem.EnableIPv6Address = TunEnableIPv6Address; |         _config.TunModeItem.EnableIPv6Address = TunEnableIPv6Address; | ||||||
|         if (TunLinuxSudoPassword != _config.TunModeItem.LinuxSudoPwd) |           | ||||||
|         { |  | ||||||
|             _config.TunModeItem.LinuxSudoPwd = DesUtils.Encrypt(TunLinuxSudoPassword); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         //coreType |         //coreType | ||||||
|         await SaveCoreType(); |         await SaveCoreType(); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -436,11 +436,13 @@ public class StatusBarViewModel : MyReactiveObject | ||||||
|                     Locator.Current.GetService<MainWindowViewModel>()?.RebootAsAdmin(); |                     Locator.Current.GetService<MainWindowViewModel>()?.RebootAsAdmin(); | ||||||
|                     return; |                     return; | ||||||
|                 } |                 } | ||||||
|                 else if (Utils.IsOSX()) |                 else | ||||||
|                 { |                 { | ||||||
|                     _config.TunModeItem.EnableTun = false; |                     if (await _updateView?.Invoke(EViewAction.PasswordInput, null) == false) | ||||||
|                     NoticeHandler.Instance.SendMessageAndEnqueue(ResUI.TbSettingsLinuxSudoPasswordIsEmpty); |                     { | ||||||
|                     return; |                         _config.TunModeItem.EnableTun = false; | ||||||
|  |                         return; | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             await ConfigHandler.SaveConfig(_config); |             await ConfigHandler.SaveConfig(_config); | ||||||
|  | @ -456,11 +458,11 @@ public class StatusBarViewModel : MyReactiveObject | ||||||
|         } |         } | ||||||
|         else if (Utils.IsLinux()) |         else if (Utils.IsLinux()) | ||||||
|         { |         { | ||||||
|             return _config.TunModeItem.LinuxSudoPwd.IsNotEmpty(); |             return AppHandler.Instance.LinuxSudoPwd.IsNotEmpty(); | ||||||
|         } |         } | ||||||
|         else if (Utils.IsOSX()) |         else if (Utils.IsOSX()) | ||||||
|         { |         { | ||||||
|             return _config.TunModeItem.LinuxSudoPwd.IsNotEmpty(); |             return AppHandler.Instance.LinuxSudoPwd.IsNotEmpty(); | ||||||
|         } |         } | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -784,28 +784,6 @@ | ||||||
|                         Margin="{StaticResource Margin4}" |                         Margin="{StaticResource Margin4}" | ||||||
|                         HorizontalAlignment="Left" /> |                         HorizontalAlignment="Left" /> | ||||||
| 
 | 
 | ||||||
|                     <TextBlock |  | ||||||
|                         x:Name="labLinuxSudoPassword" |  | ||||||
|                         Grid.Row="7" |  | ||||||
|                         Grid.Column="0" |  | ||||||
|                         Margin="{StaticResource Margin4}" |  | ||||||
|                         VerticalAlignment="Center" |  | ||||||
|                         Text="{x:Static resx:ResUI.TbSettingsLinuxSudoPassword}" /> |  | ||||||
|                     <TextBox |  | ||||||
|                         x:Name="txtLinuxSudoPassword" |  | ||||||
|                         Grid.Row="7" |  | ||||||
|                         Grid.Column="1" |  | ||||||
|                         Width="200" |  | ||||||
|                         Margin="{StaticResource Margin4}" |  | ||||||
|                         HorizontalAlignment="Left" /> |  | ||||||
|                     <TextBlock |  | ||||||
|                         x:Name="labLinuxSudoPasswordTip" |  | ||||||
|                         Grid.Row="7" |  | ||||||
|                         Grid.Column="2" |  | ||||||
|                         Margin="{StaticResource Margin4}" |  | ||||||
|                         VerticalAlignment="Center" |  | ||||||
|                         Text="{x:Static resx:ResUI.TbSettingsLinuxSudoPasswordTip}" |  | ||||||
|                         TextWrapping="Wrap" /> |  | ||||||
|                 </Grid> |                 </Grid> | ||||||
|             </TabItem> |             </TabItem> | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -153,7 +153,6 @@ public partial class OptionSettingWindow : ReactiveWindow<OptionSettingViewModel | ||||||
|             this.Bind(ViewModel, vm => vm.TunMtu, v => v.cmbMtu.SelectedValue).DisposeWith(disposables); |             this.Bind(ViewModel, vm => vm.TunMtu, v => v.cmbMtu.SelectedValue).DisposeWith(disposables); | ||||||
|             this.Bind(ViewModel, vm => vm.TunEnableExInbound, v => v.togEnableExInbound.IsChecked).DisposeWith(disposables); |             this.Bind(ViewModel, vm => vm.TunEnableExInbound, v => v.togEnableExInbound.IsChecked).DisposeWith(disposables); | ||||||
|             this.Bind(ViewModel, vm => vm.TunEnableIPv6Address, v => v.togEnableIPv6Address.IsChecked).DisposeWith(disposables); |             this.Bind(ViewModel, vm => vm.TunEnableIPv6Address, v => v.togEnableIPv6Address.IsChecked).DisposeWith(disposables); | ||||||
|             this.Bind(ViewModel, vm => vm.TunLinuxSudoPassword, v => v.txtLinuxSudoPassword.Text).DisposeWith(disposables); |  | ||||||
| 
 | 
 | ||||||
|             this.Bind(ViewModel, vm => vm.CoreType1, v => v.cmbCoreType1.SelectedValue).DisposeWith(disposables); |             this.Bind(ViewModel, vm => vm.CoreType1, v => v.cmbCoreType1.SelectedValue).DisposeWith(disposables); | ||||||
|             this.Bind(ViewModel, vm => vm.CoreType2, v => v.cmbCoreType2.SelectedValue).DisposeWith(disposables); |             this.Bind(ViewModel, vm => vm.CoreType2, v => v.cmbCoreType2.SelectedValue).DisposeWith(disposables); | ||||||
|  | @ -170,10 +169,6 @@ public partial class OptionSettingWindow : ReactiveWindow<OptionSettingViewModel | ||||||
|         { |         { | ||||||
|             txbSettingsExceptionTip2.IsVisible = false; |             txbSettingsExceptionTip2.IsVisible = false; | ||||||
| 
 | 
 | ||||||
|             txtLinuxSudoPassword.IsVisible = false; |  | ||||||
|             labLinuxSudoPassword.IsVisible = false; |  | ||||||
|             labLinuxSudoPasswordTip.IsVisible = false; |  | ||||||
| 
 |  | ||||||
|             labHide2TrayWhenClose.IsVisible = false; |             labHide2TrayWhenClose.IsVisible = false; | ||||||
|             togHide2TrayWhenClose.IsVisible = false; |             togHide2TrayWhenClose.IsVisible = false; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  | @ -4,9 +4,11 @@ using Avalonia.Controls; | ||||||
| using Avalonia.Controls.ApplicationLifetimes; | using Avalonia.Controls.ApplicationLifetimes; | ||||||
| using Avalonia.ReactiveUI; | using Avalonia.ReactiveUI; | ||||||
| using Avalonia.Threading; | using Avalonia.Threading; | ||||||
|  | using DialogHostAvalonia; | ||||||
| using ReactiveUI; | using ReactiveUI; | ||||||
| using Splat; | using Splat; | ||||||
| using v2rayN.Desktop.Common; | using v2rayN.Desktop.Common; | ||||||
|  | using static QRCoder.PayloadGenerator; | ||||||
| 
 | 
 | ||||||
| namespace v2rayN.Desktop.Views; | namespace v2rayN.Desktop.Views; | ||||||
| 
 | 
 | ||||||
|  | @ -81,6 +83,10 @@ public partial class StatusBarView : ReactiveUserControl<StatusBarViewModel> | ||||||
|                     return false; |                     return false; | ||||||
|                 await AvaUtils.SetClipboardData(this, (string)obj); |                 await AvaUtils.SetClipboardData(this, (string)obj); | ||||||
|                 break; |                 break; | ||||||
|  | 
 | ||||||
|  |             case EViewAction.PasswordInput: | ||||||
|  |                 return await PasswordInputAsync(); | ||||||
|  |                 break; | ||||||
|         } |         } | ||||||
|         return await Task.FromResult(true); |         return await Task.FromResult(true); | ||||||
|     } |     } | ||||||
|  | @ -96,6 +102,20 @@ public partial class StatusBarView : ReactiveUserControl<StatusBarViewModel> | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     private async Task<bool> PasswordInputAsync() | ||||||
|  |     { | ||||||
|  |         var dialog = new SudoPasswordInputView(); | ||||||
|  |         var obj = await DialogHost.Show(dialog); | ||||||
|  |         if (obj == null) | ||||||
|  |         { | ||||||
|  |             togEnableTun.IsChecked = false; | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         AppHandler.Instance.LinuxSudoPwd = obj.ToString() ?? string.Empty; | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     private void TxtRunningServerDisplay_Tapped(object? sender, Avalonia.Input.TappedEventArgs e) |     private void TxtRunningServerDisplay_Tapped(object? sender, Avalonia.Input.TappedEventArgs e) | ||||||
|     { |     { | ||||||
|         ViewModel?.TestServerAvailability(); |         ViewModel?.TestServerAvailability(); | ||||||
|  |  | ||||||
							
								
								
									
										70
									
								
								v2rayN/v2rayN.Desktop/Views/SudoPasswordInputView.axaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								v2rayN/v2rayN.Desktop/Views/SudoPasswordInputView.axaml
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,70 @@ | ||||||
|  | <UserControl | ||||||
|  |     x:Class="v2rayN.Desktop.Views.SudoPasswordInputView" | ||||||
|  |     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" | ||||||
|  |     d:DesignHeight="200" | ||||||
|  |     d:DesignWidth="400" | ||||||
|  |     mc:Ignorable="d"> | ||||||
|  |     <DockPanel Margin="{StaticResource Margin8}"> | ||||||
|  | 
 | ||||||
|  |         <Border | ||||||
|  |             Margin="{StaticResource Margin4}" | ||||||
|  |             VerticalAlignment="Center" | ||||||
|  |             DockPanel.Dock="Bottom" | ||||||
|  |             Theme="{DynamicResource CardBorder}"> | ||||||
|  | 
 | ||||||
|  |             <StackPanel | ||||||
|  |                 Margin="{StaticResource Margin4}" | ||||||
|  |                 HorizontalAlignment="Center" | ||||||
|  |                 Orientation="Horizontal"> | ||||||
|  |                 <Button | ||||||
|  |                     x:Name="btnSave" | ||||||
|  |                     Width="100" | ||||||
|  |                     Content="{x:Static resx:ResUI.TbConfirm}" | ||||||
|  |                     Cursor="Hand" | ||||||
|  |                     IsDefault="True" /> | ||||||
|  |                 <Button | ||||||
|  |                     x:Name="btnCancel" | ||||||
|  |                     Width="100" | ||||||
|  |                     Margin="{StaticResource MarginLr8}" | ||||||
|  |                     Content="{x:Static resx:ResUI.TbCancel}" | ||||||
|  |                     Cursor="Hand" | ||||||
|  |                     IsCancel="True" /> | ||||||
|  |             </StackPanel> | ||||||
|  |         </Border> | ||||||
|  | 
 | ||||||
|  |         <Border | ||||||
|  |             Margin="{StaticResource Margin4}" | ||||||
|  |             VerticalAlignment="Center" | ||||||
|  |             Theme="{DynamicResource CardBorder}"> | ||||||
|  | 
 | ||||||
|  |             <Grid ColumnDefinitions="Auto,400" RowDefinitions="Auto,Auto,Auto"> | ||||||
|  |                 <TextBlock | ||||||
|  |                     Grid.Row="1" | ||||||
|  |                     Grid.Column="0" | ||||||
|  |                     Margin="{StaticResource Margin4}" | ||||||
|  |                     VerticalAlignment="Center" | ||||||
|  |                     Text="{x:Static resx:ResUI.TbSettingsLinuxSudoPassword}" /> | ||||||
|  | 
 | ||||||
|  |                 <TextBox | ||||||
|  |                     x:Name="txtPassword" | ||||||
|  |                     Grid.Row="1" | ||||||
|  |                     Grid.Column="1" | ||||||
|  |                     Margin="{StaticResource Margin4}" | ||||||
|  |                     Classes="revealPasswordButton" | ||||||
|  |                     Focusable="True" /> | ||||||
|  | 
 | ||||||
|  |                 <TextBlock | ||||||
|  |                     Grid.Row="2" | ||||||
|  |                     Grid.Column="1" | ||||||
|  |                     Margin="{StaticResource Margin4}" | ||||||
|  |                     VerticalAlignment="Center" | ||||||
|  |                     Text="{x:Static resx:ResUI.TbSettingsLinuxSudoPasswordTip}" | ||||||
|  |                     TextWrapping="Wrap" /> | ||||||
|  |             </Grid> | ||||||
|  |         </Border> | ||||||
|  |     </DockPanel> | ||||||
|  | </UserControl> | ||||||
							
								
								
									
										33
									
								
								v2rayN/v2rayN.Desktop/Views/SudoPasswordInputView.axaml.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								v2rayN/v2rayN.Desktop/Views/SudoPasswordInputView.axaml.cs
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,33 @@ | ||||||
|  | using Avalonia.Controls; | ||||||
|  | using Avalonia.Threading; | ||||||
|  | using DialogHostAvalonia; | ||||||
|  | 
 | ||||||
|  | namespace v2rayN.Desktop.Views; | ||||||
|  | 
 | ||||||
|  | public partial class SudoPasswordInputView : UserControl | ||||||
|  | { | ||||||
|  |     public SudoPasswordInputView() | ||||||
|  |     { | ||||||
|  |         InitializeComponent(); | ||||||
|  | 
 | ||||||
|  |         this.Loaded += (s, e) => txtPassword.Focus(); | ||||||
|  | 
 | ||||||
|  |         btnSave.Click += (_, _) => | ||||||
|  |         { | ||||||
|  |             if (string.IsNullOrEmpty(txtPassword.Text)) | ||||||
|  |             { | ||||||
|  |                 txtPassword.Focus(); | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  |             Dispatcher.UIThread.Post(() => | ||||||
|  |             { | ||||||
|  |                 DialogHost.Close(null, txtPassword.Text); | ||||||
|  |             }); | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         btnCancel.Click += (_, _) => | ||||||
|  |         { | ||||||
|  |             DialogHost.Close(null); | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  | } | ||||||
		Loading…
	
		Reference in a new issue
	
	 2dust
						2dust