Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Adding a compatible layout for RTL cultures #16481

Closed
Closed
2 changes: 2 additions & 0 deletions src/Files.App.CsWin32/NativeMethods.txt
Original file line number Diff line number Diff line change
Expand Up @@ -156,3 +156,5 @@ IApplicationDestinations
ApplicationDestinations
IApplicationDocumentLists
ApplicationDocumentLists
SetWindowLongPtr
GetWindowLongPtr
1 change: 1 addition & 0 deletions src/Files.App/Dialogs/AddBranchDialog.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ public AddBranchDialogViewModel ViewModel
public AddBranchDialog()
{
InitializeComponent();
AppLanguageHelper.UpdateContextLayout(this);
}

public new async Task<DialogResult> ShowAsync() => (DialogResult)await base.ShowAsync();
Expand Down
1 change: 1 addition & 0 deletions src/Files.App/Dialogs/AddItemDialog.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ public AddItemDialogViewModel ViewModel
public AddItemDialog()
{
InitializeComponent();
AppLanguageHelper.UpdateContextLayout(this);
}

public new async Task<DialogResult> ShowAsync()
Expand Down
1 change: 1 addition & 0 deletions src/Files.App/Dialogs/BulkRenameDialog.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ public BulkRenameDialogViewModel ViewModel
public BulkRenameDialog()
{
InitializeComponent();
AppLanguageHelper.UpdateContextLayout(this);
}

public new async Task<DialogResult> ShowAsync()
Expand Down
1 change: 1 addition & 0 deletions src/Files.App/Dialogs/CreateArchiveDialog.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -65,6 +65,7 @@ public CreateArchiveDialog()
InitializeComponent();

ViewModel.PropertyChanged += ViewModel_PropertyChanged;
AppLanguageHelper.UpdateContextLayout(this);
}

public new Task<ContentDialogResult> ShowAsync()
Expand Down
1 change: 1 addition & 0 deletions src/Files.App/Dialogs/CreateShortcutDialog.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ public CreateShortcutDialog()
{
Source = ShortcutTarget
});
AppLanguageHelper.UpdateContextLayout(this);
}

private void CreateShortcutDialog_Closing(ContentDialog sender, ContentDialogClosingEventArgs args)
Expand Down
1 change: 1 addition & 0 deletions src/Files.App/Dialogs/CredentialDialog.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ public CredentialDialogViewModel ViewModel
public CredentialDialog()
{
InitializeComponent();
AppLanguageHelper.UpdateContextLayout(this);
}

public new async Task<DialogResult> ShowAsync()
Expand Down
1 change: 1 addition & 0 deletions src/Files.App/Dialogs/DecompressArchiveDialog.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ public DecompressArchiveDialogViewModel ViewModel
public DecompressArchiveDialog()
{
InitializeComponent();
AppLanguageHelper.UpdateContextLayout(this);
}

private void ContentDialog_PrimaryButtonClick(ContentDialog sender, ContentDialogButtonClickEventArgs args)
Expand Down
1 change: 1 addition & 0 deletions src/Files.App/Dialogs/DynamicDialog.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ public DynamicDialog(DynamicDialogViewModel dynamicDialogViewModel)

dynamicDialogViewModel.HideDialog = Hide;
ViewModel = dynamicDialogViewModel;
AppLanguageHelper.UpdateContextLayout(this);
}

private void ContentDialog_PrimaryButtonClick(ContentDialog sender, ContentDialogButtonClickEventArgs args)
Expand Down
1 change: 1 addition & 0 deletions src/Files.App/Dialogs/ElevateConfirmDialog.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ public ElevateConfirmDialogViewModel ViewModel
public ElevateConfirmDialog()
{
InitializeComponent();
AppLanguageHelper.UpdateContextLayout(this);
}

public new async Task<DialogResult> ShowAsync()
Expand Down
1 change: 1 addition & 0 deletions src/Files.App/Dialogs/FileTooLargeDialog.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ public FileTooLargeDialogViewModel ViewModel
public FileTooLargeDialog()
{
InitializeComponent();
AppLanguageHelper.UpdateContextLayout(this);
}

public new async Task<DialogResult> ShowAsync()
Expand Down
1 change: 1 addition & 0 deletions src/Files.App/Dialogs/FilesystemOperationDialog.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@ public FilesystemOperationDialog()
InitializeComponent();

MainWindow.Instance.SizeChanged += Current_SizeChanged;
AppLanguageHelper.UpdateContextLayout(this);
}

public new async Task<DialogResult> ShowAsync()
Expand Down
1 change: 1 addition & 0 deletions src/Files.App/Dialogs/GitHubLoginDialog.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ public GitHubLoginDialogViewModel ViewModel
public GitHubLoginDialog()
{
InitializeComponent();
AppLanguageHelper.UpdateContextLayout(this);
}

public new async Task<DialogResult> ShowAsync()
Expand Down
1 change: 1 addition & 0 deletions src/Files.App/Dialogs/ReleaseNotesDialog.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ public ReleaseNotesDialog()
InitializeComponent();

MainWindow.Instance.SizeChanged += Current_SizeChanged;
AppLanguageHelper.UpdateContextLayout(this);
UpdateDialogLayout();
}

Expand Down
1 change: 1 addition & 0 deletions src/Files.App/Dialogs/ReorderSidebarItemsDialog.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ public ReorderSidebarItemsDialogViewModel ViewModel
public ReorderSidebarItemsDialog()
{
InitializeComponent();
AppLanguageHelper.UpdateContextLayout(this);
}

private async void MoveItemAsync(object sender, PointerRoutedEventArgs e)
Expand Down
1 change: 1 addition & 0 deletions src/Files.App/Dialogs/SettingsDialog.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ public SettingsDialog()
InitializeComponent();

MainWindow.Instance.SizeChanged += Current_SizeChanged;
AppLanguageHelper.UpdateContextLayout(this);
UpdateDialogLayout();
}

Expand Down
64 changes: 60 additions & 4 deletions src/Files.App/Helpers/Application/AppLanguageHelper.cs
Original file line number Diff line number Diff line change
@@ -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
/// <summary>
/// Provides static helper to manage supported languages in the application.
/// </summary>
Expand All @@ -27,6 +34,10 @@ public static class AppLanguageHelper
/// </summary>
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;

/// <summary>
/// Initializes the <see cref="AppLanguageHelper"/> class.
/// </summary>
Expand All @@ -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);
Expand Down Expand Up @@ -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
/// <summary>
/// Updates the title bar layout of the specified window based on the current culture.
/// </summary>
/// <param name="window">The window to be updated.</param>
/// <returns>True if the update was successful; otherwise, false.</returns>
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;
}
}
}
2 changes: 1 addition & 1 deletion src/Files.App/Helpers/Win32/Win32PInvoke.Methods.cs
Original file line number Diff line number Diff line change
@@ -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;
Expand Down
3 changes: 3 additions & 0 deletions src/Files.App/MainWindow.xaml.cs
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,8 @@ public void ShowSplashScreen()
var rootFrame = EnsureWindowIsInitialized();

rootFrame?.Navigate(typeof(SplashScreenPage));

AppLanguageHelper.UpdateTitleBar(Instance);
}

public async Task InitializeApplicationAsync(object activatedEventArgs)
Expand Down Expand Up @@ -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()
Expand Down
133 changes: 76 additions & 57 deletions src/Files.App/UserControls/AddressToolbar.xaml
Original file line number Diff line number Diff line change
Expand Up @@ -263,63 +263,68 @@
</AnimatedIcon>
</ToggleButton>

<Button
x:Name="Back"
AccessKey="B"
AccessKeyInvoked="Button_AccessKeyInvoked"
AutomationProperties.FullDescription="{x:Bind ViewModel.Commands.NavigateBack.Description, Mode=OneWay}"
AutomationProperties.Name="{x:Bind ViewModel.Commands.NavigateBack.Label, Mode=OneWay}"
Command="{x:Bind ViewModel.Commands.NavigateBack, Mode=OneWay}"
IsEnabled="{x:Bind ViewModel.Commands.NavigateBack.IsExecutable, Mode=OneWay}"
Style="{StaticResource AddressToolbarButtonStyle}"
ToolTipService.ToolTip="{x:Bind ViewModel.Commands.NavigateBack.LabelWithHotKey, Mode=OneWay}">
<FontIcon FontSize="14" Glyph="{x:Bind ViewModel.Commands.NavigateBack.Glyph.BaseGlyph, Mode=OneWay}" />
<Button.ContextFlyout>
<MenuFlyout
x:Name="BackHistoryFlyout"
Opening="BackHistoryFlyout_Opening"
Placement="BottomEdgeAlignedLeft"
ScrollViewer.VerticalScrollBarVisibility="Auto"
ScrollViewer.VerticalScrollMode="Auto">
<MenuFlyout.MenuFlyoutPresenterStyle>
<Style TargetType="MenuFlyoutPresenter">
<Setter Property="MaxHeight" Value="400" />
<!-- Workaround for https://github.com/files-community/Files/issues/13078 -->
<Setter Target="HighContrastAdjustment" Value="None" />
</Style>
</MenuFlyout.MenuFlyoutPresenterStyle>
</MenuFlyout>
</Button.ContextFlyout>
</Button>

<Button
x:Name="Forward"
AccessKey="F"
AccessKeyInvoked="Button_AccessKeyInvoked"
AutomationProperties.FullDescription="{x:Bind ViewModel.Commands.NavigateForward.Description, Mode=OneWay}"
AutomationProperties.Name="{x:Bind ViewModel.Commands.NavigateForward.Label, Mode=OneWay}"
Command="{x:Bind ViewModel.Commands.NavigateForward, Mode=OneWay}"
IsEnabled="{x:Bind ViewModel.Commands.NavigateForward.IsExecutable, Mode=OneWay}"
Style="{StaticResource AddressToolbarButtonStyle}"
ToolTipService.ToolTip="{x:Bind ViewModel.Commands.NavigateForward.LabelWithHotKey, Mode=OneWay}">
<FontIcon FontSize="14" Glyph="{x:Bind ViewModel.Commands.NavigateForward.Glyph.BaseGlyph, Mode=OneWay}" />
<Button.ContextFlyout>
<MenuFlyout
x:Name="ForwardHistoryFlyout"
Opening="ForwardHistoryFlyout_Opening"
Placement="BottomEdgeAlignedLeft"
ScrollViewer.VerticalScrollBarVisibility="Auto"
ScrollViewer.VerticalScrollMode="Auto">
<MenuFlyout.MenuFlyoutPresenterStyle>
<Style TargetType="MenuFlyoutPresenter">
<Setter Property="MaxHeight" Value="400" />
<!-- Workaround for https://github.com/files-community/Files/issues/13078 -->
<Setter Target="HighContrastAdjustment" Value="None" />
</Style>
</MenuFlyout.MenuFlyoutPresenterStyle>
</MenuFlyout>
</Button.ContextFlyout>
</Button>
<StackPanel
x:Name="NavigationButtons"
FlowDirection="{Binding Path=FlowDirection, ElementName=Root}"
Orientation="Horizontal">
<Button
x:Name="Back"
AccessKey="B"
AccessKeyInvoked="Button_AccessKeyInvoked"
AutomationProperties.FullDescription="{x:Bind ViewModel.Commands.NavigateBack.Description, Mode=OneWay}"
AutomationProperties.Name="{x:Bind ViewModel.Commands.NavigateBack.Label, Mode=OneWay}"
Command="{x:Bind ViewModel.Commands.NavigateBack, Mode=OneWay}"
IsEnabled="{x:Bind ViewModel.Commands.NavigateBack.IsExecutable, Mode=OneWay}"
Style="{StaticResource AddressToolbarButtonStyle}"
ToolTipService.ToolTip="{x:Bind ViewModel.Commands.NavigateBack.LabelWithHotKey, Mode=OneWay}">
<FontIcon FontSize="14" Glyph="{x:Bind ViewModel.Commands.NavigateBack.Glyph.BaseGlyph, Mode=OneWay}" />
<Button.ContextFlyout>
<MenuFlyout
x:Name="BackHistoryFlyout"
Opening="BackHistoryFlyout_Opening"
Placement="BottomEdgeAlignedLeft"
ScrollViewer.VerticalScrollBarVisibility="Auto"
ScrollViewer.VerticalScrollMode="Auto">
<MenuFlyout.MenuFlyoutPresenterStyle>
<Style TargetType="MenuFlyoutPresenter">
<Setter Property="MaxHeight" Value="400" />
<!-- Workaround for https://github.com/files-community/Files/issues/13078 -->
<Setter Target="HighContrastAdjustment" Value="None" />
</Style>
</MenuFlyout.MenuFlyoutPresenterStyle>
</MenuFlyout>
</Button.ContextFlyout>
</Button>

<Button
x:Name="Forward"
AccessKey="F"
AccessKeyInvoked="Button_AccessKeyInvoked"
AutomationProperties.FullDescription="{x:Bind ViewModel.Commands.NavigateForward.Description, Mode=OneWay}"
AutomationProperties.Name="{x:Bind ViewModel.Commands.NavigateForward.Label, Mode=OneWay}"
Command="{x:Bind ViewModel.Commands.NavigateForward, Mode=OneWay}"
IsEnabled="{x:Bind ViewModel.Commands.NavigateForward.IsExecutable, Mode=OneWay}"
Style="{StaticResource AddressToolbarButtonStyle}"
ToolTipService.ToolTip="{x:Bind ViewModel.Commands.NavigateForward.LabelWithHotKey, Mode=OneWay}">
<FontIcon FontSize="14" Glyph="{x:Bind ViewModel.Commands.NavigateForward.Glyph.BaseGlyph, Mode=OneWay}" />
<Button.ContextFlyout>
<MenuFlyout
x:Name="ForwardHistoryFlyout"
Opening="ForwardHistoryFlyout_Opening"
Placement="BottomEdgeAlignedLeft"
ScrollViewer.VerticalScrollBarVisibility="Auto"
ScrollViewer.VerticalScrollMode="Auto">
<MenuFlyout.MenuFlyoutPresenterStyle>
<Style TargetType="MenuFlyoutPresenter">
<Setter Property="MaxHeight" Value="400" />
<!-- Workaround for https://github.com/files-community/Files/issues/13078 -->
<Setter Target="HighContrastAdjustment" Value="None" />
</Style>
</MenuFlyout.MenuFlyoutPresenterStyle>
</MenuFlyout>
</Button.ContextFlyout>
</Button>
</StackPanel>

<Button
x:Name="Up"
Expand Down Expand Up @@ -691,6 +696,20 @@
<Setter Target="StatusInfoBadge.Style" Value="{StaticResource CriticalIconInfoBadgeStyle}" />
</VisualState.Setters>
</VisualState>
<!-- FlowDirection LeftToRight -->
<VisualState x:Name="LTR">
<VisualState.Setters>
<Setter Target="NavigationButtons.Children[0]" Value="Back" />
<Setter Target="NavigationButtons.Children[1]" Value="Forward" />
</VisualState.Setters>
</VisualState>
<!-- FlowDirection RightToLeft -->
<VisualState x:Name="RTL">
<VisualState.Setters>
<Setter Target="NavigationButtons.Children[0]" Value="Forward" />
<Setter Target="NavigationButtons.Children[1]" Value="Back" />
</VisualState.Setters>
</VisualState>
</VisualStateGroup>
</VisualStateManager.VisualStateGroups>
</Grid>
Expand Down
Loading