diff --git a/src/Files.App.CsWin32/NativeMethods.txt b/src/Files.App.CsWin32/NativeMethods.txt index 3b611b024af5..72c18b60ebf5 100644 --- a/src/Files.App.CsWin32/NativeMethods.txt +++ b/src/Files.App.CsWin32/NativeMethods.txt @@ -156,3 +156,5 @@ IApplicationDestinations ApplicationDestinations IApplicationDocumentLists ApplicationDocumentLists +SetWindowLongPtr +GetWindowLongPtr \ No newline at end of file diff --git a/src/Files.App/Dialogs/AddBranchDialog.xaml.cs b/src/Files.App/Dialogs/AddBranchDialog.xaml.cs index 0087cbab901b..a35c339bc8b3 100644 --- a/src/Files.App/Dialogs/AddBranchDialog.xaml.cs +++ b/src/Files.App/Dialogs/AddBranchDialog.xaml.cs @@ -21,6 +21,7 @@ public AddBranchDialogViewModel ViewModel public AddBranchDialog() { InitializeComponent(); + AppLanguageHelper.UpdateContextLayout(this); } public new async Task ShowAsync() => (DialogResult)await base.ShowAsync(); diff --git a/src/Files.App/Dialogs/AddItemDialog.xaml.cs b/src/Files.App/Dialogs/AddItemDialog.xaml.cs index 938f17d42495..4033cb52c3d3 100644 --- a/src/Files.App/Dialogs/AddItemDialog.xaml.cs +++ b/src/Files.App/Dialogs/AddItemDialog.xaml.cs @@ -24,6 +24,7 @@ public AddItemDialogViewModel ViewModel public AddItemDialog() { InitializeComponent(); + AppLanguageHelper.UpdateContextLayout(this); } public new async Task ShowAsync() diff --git a/src/Files.App/Dialogs/BulkRenameDialog.xaml.cs b/src/Files.App/Dialogs/BulkRenameDialog.xaml.cs index a7f5b5478b65..f442aeda3b89 100644 --- a/src/Files.App/Dialogs/BulkRenameDialog.xaml.cs +++ b/src/Files.App/Dialogs/BulkRenameDialog.xaml.cs @@ -19,6 +19,7 @@ public BulkRenameDialogViewModel ViewModel public BulkRenameDialog() { InitializeComponent(); + AppLanguageHelper.UpdateContextLayout(this); } public new async Task ShowAsync() diff --git a/src/Files.App/Dialogs/CreateArchiveDialog.xaml.cs b/src/Files.App/Dialogs/CreateArchiveDialog.xaml.cs index f54d0227afbc..fc58041cc3c0 100644 --- a/src/Files.App/Dialogs/CreateArchiveDialog.xaml.cs +++ b/src/Files.App/Dialogs/CreateArchiveDialog.xaml.cs @@ -65,6 +65,7 @@ public CreateArchiveDialog() InitializeComponent(); ViewModel.PropertyChanged += ViewModel_PropertyChanged; + AppLanguageHelper.UpdateContextLayout(this); } public new Task ShowAsync() diff --git a/src/Files.App/Dialogs/CreateShortcutDialog.xaml.cs b/src/Files.App/Dialogs/CreateShortcutDialog.xaml.cs index 09cd2f963e9f..1d5d88183573 100644 --- a/src/Files.App/Dialogs/CreateShortcutDialog.xaml.cs +++ b/src/Files.App/Dialogs/CreateShortcutDialog.xaml.cs @@ -29,6 +29,7 @@ public CreateShortcutDialog() { Source = ShortcutTarget }); + AppLanguageHelper.UpdateContextLayout(this); } private void CreateShortcutDialog_Closing(ContentDialog sender, ContentDialogClosingEventArgs args) diff --git a/src/Files.App/Dialogs/CredentialDialog.xaml.cs b/src/Files.App/Dialogs/CredentialDialog.xaml.cs index 92ba79ba03c5..c0fa59e38bf7 100644 --- a/src/Files.App/Dialogs/CredentialDialog.xaml.cs +++ b/src/Files.App/Dialogs/CredentialDialog.xaml.cs @@ -21,6 +21,7 @@ public CredentialDialogViewModel ViewModel public CredentialDialog() { InitializeComponent(); + AppLanguageHelper.UpdateContextLayout(this); } public new async Task ShowAsync() diff --git a/src/Files.App/Dialogs/DecompressArchiveDialog.xaml.cs b/src/Files.App/Dialogs/DecompressArchiveDialog.xaml.cs index 7a4543fd8476..748bbce6a294 100644 --- a/src/Files.App/Dialogs/DecompressArchiveDialog.xaml.cs +++ b/src/Files.App/Dialogs/DecompressArchiveDialog.xaml.cs @@ -22,6 +22,7 @@ public DecompressArchiveDialogViewModel ViewModel public DecompressArchiveDialog() { InitializeComponent(); + AppLanguageHelper.UpdateContextLayout(this); } private void ContentDialog_PrimaryButtonClick(ContentDialog sender, ContentDialogButtonClickEventArgs args) diff --git a/src/Files.App/Dialogs/DynamicDialog.xaml.cs b/src/Files.App/Dialogs/DynamicDialog.xaml.cs index fe0f628440a7..fd3788f5f7de 100644 --- a/src/Files.App/Dialogs/DynamicDialog.xaml.cs +++ b/src/Files.App/Dialogs/DynamicDialog.xaml.cs @@ -34,6 +34,7 @@ public DynamicDialog(DynamicDialogViewModel dynamicDialogViewModel) dynamicDialogViewModel.HideDialog = Hide; ViewModel = dynamicDialogViewModel; + AppLanguageHelper.UpdateContextLayout(this); } private void ContentDialog_PrimaryButtonClick(ContentDialog sender, ContentDialogButtonClickEventArgs args) diff --git a/src/Files.App/Dialogs/ElevateConfirmDialog.xaml.cs b/src/Files.App/Dialogs/ElevateConfirmDialog.xaml.cs index 924602ce2b3d..a51822b71d1a 100644 --- a/src/Files.App/Dialogs/ElevateConfirmDialog.xaml.cs +++ b/src/Files.App/Dialogs/ElevateConfirmDialog.xaml.cs @@ -23,6 +23,7 @@ public ElevateConfirmDialogViewModel ViewModel public ElevateConfirmDialog() { InitializeComponent(); + AppLanguageHelper.UpdateContextLayout(this); } public new async Task ShowAsync() diff --git a/src/Files.App/Dialogs/FileTooLargeDialog.xaml.cs b/src/Files.App/Dialogs/FileTooLargeDialog.xaml.cs index 4daf884822b9..9dc0bb93fdb9 100644 --- a/src/Files.App/Dialogs/FileTooLargeDialog.xaml.cs +++ b/src/Files.App/Dialogs/FileTooLargeDialog.xaml.cs @@ -20,6 +20,7 @@ public FileTooLargeDialogViewModel ViewModel public FileTooLargeDialog() { InitializeComponent(); + AppLanguageHelper.UpdateContextLayout(this); } public new async Task ShowAsync() diff --git a/src/Files.App/Dialogs/FilesystemOperationDialog.xaml.cs b/src/Files.App/Dialogs/FilesystemOperationDialog.xaml.cs index 9760cfa332dd..a9984e9bea6a 100644 --- a/src/Files.App/Dialogs/FilesystemOperationDialog.xaml.cs +++ b/src/Files.App/Dialogs/FilesystemOperationDialog.xaml.cs @@ -32,6 +32,7 @@ public FilesystemOperationDialog() InitializeComponent(); MainWindow.Instance.SizeChanged += Current_SizeChanged; + AppLanguageHelper.UpdateContextLayout(this); } public new async Task ShowAsync() diff --git a/src/Files.App/Dialogs/GitHubLoginDialog.xaml.cs b/src/Files.App/Dialogs/GitHubLoginDialog.xaml.cs index 973909dec988..dfc13fcd1a53 100644 --- a/src/Files.App/Dialogs/GitHubLoginDialog.xaml.cs +++ b/src/Files.App/Dialogs/GitHubLoginDialog.xaml.cs @@ -29,6 +29,7 @@ public GitHubLoginDialogViewModel ViewModel public GitHubLoginDialog() { InitializeComponent(); + AppLanguageHelper.UpdateContextLayout(this); } public new async Task ShowAsync() diff --git a/src/Files.App/Dialogs/ReleaseNotesDialog.xaml.cs b/src/Files.App/Dialogs/ReleaseNotesDialog.xaml.cs index d5aae7b549ea..adccfb25585c 100644 --- a/src/Files.App/Dialogs/ReleaseNotesDialog.xaml.cs +++ b/src/Files.App/Dialogs/ReleaseNotesDialog.xaml.cs @@ -23,6 +23,7 @@ public ReleaseNotesDialog() InitializeComponent(); MainWindow.Instance.SizeChanged += Current_SizeChanged; + AppLanguageHelper.UpdateContextLayout(this); UpdateDialogLayout(); } diff --git a/src/Files.App/Dialogs/ReorderSidebarItemsDialog.xaml.cs b/src/Files.App/Dialogs/ReorderSidebarItemsDialog.xaml.cs index 598d1da9edf6..9ccbe90915f4 100644 --- a/src/Files.App/Dialogs/ReorderSidebarItemsDialog.xaml.cs +++ b/src/Files.App/Dialogs/ReorderSidebarItemsDialog.xaml.cs @@ -29,6 +29,7 @@ public ReorderSidebarItemsDialogViewModel ViewModel public ReorderSidebarItemsDialog() { InitializeComponent(); + AppLanguageHelper.UpdateContextLayout(this); } private async void MoveItemAsync(object sender, PointerRoutedEventArgs e) diff --git a/src/Files.App/Dialogs/SettingsDialog.xaml.cs b/src/Files.App/Dialogs/SettingsDialog.xaml.cs index e70582a22432..c63a46bc62b8 100644 --- a/src/Files.App/Dialogs/SettingsDialog.xaml.cs +++ b/src/Files.App/Dialogs/SettingsDialog.xaml.cs @@ -24,6 +24,7 @@ public SettingsDialog() InitializeComponent(); MainWindow.Instance.SizeChanged += Current_SizeChanged; + AppLanguageHelper.UpdateContextLayout(this); UpdateDialogLayout(); } diff --git a/src/Files.App/Helpers/Application/AppLanguageHelper.cs b/src/Files.App/Helpers/Application/AppLanguageHelper.cs index 4a70d84fec66..a414c8a62767 100644 --- a/src/Files.App/Helpers/Application/AppLanguageHelper.cs +++ b/src/Files.App/Helpers/Application/AppLanguageHelper.cs @@ -1,11 +1,18 @@ // Copyright (c) 2024 Files Community // Licensed under the MIT License. See the LICENSE. +using Microsoft.UI.Xaml; using System.Globalization; +using System.Text.RegularExpressions; using Windows.Globalization; +using WinRT.Interop; +using Windows.Win32; +using Windows.Win32.Foundation; +using Windows.Win32.UI.WindowsAndMessaging; namespace Files.App.Helpers { + // TODO: Replaced by RealTime Resources in the future /// /// Provides static helper to manage supported languages in the application. /// @@ -27,6 +34,10 @@ public static class AppLanguageHelper /// public static AppLanguageItem PreferredLanguage { get; private set; } + public static CultureInfo PreferredCulture => new(PreferredLanguage.Code); + + public static FlowDirection FlowDirection => PreferredCulture.TextInfo.IsRightToLeft ? FlowDirection.RightToLeft : FlowDirection.LeftToRight; + /// /// Initializes the class. /// @@ -48,10 +59,10 @@ static AppLanguageHelper() // Set the system default language as the first item in the Languages collection var systemLanguage = new AppLanguageItem(CultureInfo.InstalledUICulture.Name, systemDefault: true); - if (appLanguages.Select(lang => lang.Name.Contains(systemLanguage.Name)).Any()) - appLanguages[0] = systemLanguage; - else - appLanguages[0] = new("en-US", systemDefault: true); + + appLanguages[0] = appLanguages.Select(lang => lang.Name.Contains(systemLanguage.Name)).Any() + ? systemLanguage + : new("en-US", systemDefault: true); // Initialize the list SupportedLanguages = new(appLanguages); @@ -104,5 +115,50 @@ public static bool TryChange(string code) ApplicationLanguages.PrimaryLanguageOverride = index == 0 ? _defaultCode : PreferredLanguage.Code; return true; } + + // TODO: Replaced by RealTime Resources in the future + /// + /// Updates the title bar layout of the specified window based on the current culture. + /// + /// The window to be updated. + /// True if the update was successful; otherwise, false. + public static bool UpdateTitleBar(Window window) + { + try + { + var hwnd = new HWND(WindowNative.GetWindowHandle(window)); + var exStyle = PInvoke.GetWindowLongPtr(hwnd, WINDOW_LONG_PTR_INDEX.GWL_EXSTYLE); + + exStyle = PreferredCulture.TextInfo.IsRightToLeft + ? new((uint)exStyle | (uint)WINDOW_EX_STYLE.WS_EX_LAYOUTRTL) // Set RTL layout + : new((uint)exStyle.ToInt64() & ~(uint)WINDOW_EX_STYLE.WS_EX_LAYOUTRTL); // Set LTR layout + + if (PInvoke.SetWindowLongPtr(hwnd, WINDOW_LONG_PTR_INDEX.GWL_EXSTYLE, exStyle) == 0) + return false; + } + catch (Exception) + { + return false; + } + + return true; + } + + // TODO: Replaced by RealTime Resources in the future + public static void UpdateContextLayout(FrameworkElement element) + { + element.FlowDirection = FlowDirection; + } + + // TODO: Replaced by RealTime Resources in the future + public static bool SetCultureLayout(Window window) + { + var res = UpdateTitleBar(window); + if (!res) + return res; + if (window.Content is FrameworkElement element) + UpdateContextLayout(element); + return res; + } } } diff --git a/src/Files.App/Helpers/Win32/Win32PInvoke.Methods.cs b/src/Files.App/Helpers/Win32/Win32PInvoke.Methods.cs index e6f07783d534..a9a557e3a973 100644 --- a/src/Files.App/Helpers/Win32/Win32PInvoke.Methods.cs +++ b/src/Files.App/Helpers/Win32/Win32PInvoke.Methods.cs @@ -1,4 +1,4 @@ -// Copyright (c) 2024 Files Community +// Copyright (c) 2024 Files Community // Licensed under the MIT License. See the LICENSE. using System.IO; diff --git a/src/Files.App/MainWindow.xaml.cs b/src/Files.App/MainWindow.xaml.cs index e80e55247649..d4a4ea29b72f 100644 --- a/src/Files.App/MainWindow.xaml.cs +++ b/src/Files.App/MainWindow.xaml.cs @@ -42,6 +42,8 @@ public void ShowSplashScreen() var rootFrame = EnsureWindowIsInitialized(); rootFrame?.Navigate(typeof(SplashScreenPage)); + + AppLanguageHelper.UpdateTitleBar(Instance); } public async Task InitializeApplicationAsync(object activatedEventArgs) @@ -196,6 +198,7 @@ public async Task InitializeApplicationAsync(object activatedEventArgs) if (Windows.Win32.PInvoke.IsIconic(new(WindowHandle))) WinUIEx.WindowExtensions.Restore(Instance); // Restore window if minimized + AppLanguageHelper.UpdateContextLayout(rootFrame); } private Frame? EnsureWindowIsInitialized() diff --git a/src/Files.App/UserControls/AddressToolbar.xaml b/src/Files.App/UserControls/AddressToolbar.xaml index fcf48e0c9648..e43cd78af2a2 100644 --- a/src/Files.App/UserControls/AddressToolbar.xaml +++ b/src/Files.App/UserControls/AddressToolbar.xaml @@ -263,63 +263,68 @@ - - - + + + + +