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

Feature: Added open in VS/VS Code to status bar #12645

Merged
merged 17 commits into from
Jun 19, 2023
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
54 changes: 54 additions & 0 deletions src/Files.App/Actions/Open/OpenInVSAction.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
using Files.App.Contexts;
yaira2 marked this conversation as resolved.
Show resolved Hide resolved
using Files.App.Shell;
using Microsoft.Win32;

namespace Files.App.Actions
{
internal class OpenInVSAction : ObservableObject, IAction
{
private readonly IContentPageContext _context;

private readonly bool _isVSInstalled;

public string Label { get; } = "OpenInVS".GetLocalizedResource();

public string Description { get; } = "OpenInVSDescription".GetLocalizedResource();

public bool IsExecutable =>
_isVSInstalled &&
!string.IsNullOrWhiteSpace(_context.SolutionFilePath);

public OpenInVSAction()
{
_context = Ioc.Default.GetRequiredService<IContentPageContext>();

_isVSInstalled = IsVSInstalled();
if (_isVSInstalled )
_context.PropertyChanged += Context_PropertyChanged;
}

private void Context_PropertyChanged(object? sender, PropertyChangedEventArgs e)
{
if (e.PropertyName == nameof(IContentPageContext.SolutionFilePath))
OnPropertyChanged(nameof(IsExecutable));
}

public Task ExecuteAsync()
{
Win32API.RunPowershellCommand($"start {_context.SolutionFilePath}", false);
return Task.CompletedTask;
ferrariofilippo marked this conversation as resolved.
Show resolved Hide resolved
}

private static bool IsVSInstalled()
yaira2 marked this conversation as resolved.
Show resolved Hide resolved
{
string registryKey = @"SOFTWARE\Microsoft\VisualStudio";

var key = Registry.LocalMachine.OpenSubKey(registryKey);
if (key is null)
return false;

key.Close();
return true;
}
}
}
67 changes: 67 additions & 0 deletions src/Files.App/Actions/Open/OpenInVSCodeAction.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
using Files.App.Contexts;
yaira2 marked this conversation as resolved.
Show resolved Hide resolved
using Files.App.Shell;
using Microsoft.Win32;

namespace Files.App.Actions
{
internal class OpenInVSCodeAction : ObservableObject, IAction
{
private readonly IContentPageContext _context;

private readonly bool _isVSCodeInstalled;

public string Label { get; } = "OpenInVSCode".GetLocalizedResource();

public string Description { get; } = "OpenInVSCodeDescription".GetLocalizedResource();

public bool IsExecutable =>
_isVSCodeInstalled &&
(_context.ShellPage?.InstanceViewModel.IsGitRepository ?? false);

public OpenInVSCodeAction()
{
_context = Ioc.Default.GetRequiredService<IContentPageContext>();

_isVSCodeInstalled = IsVSCodeInstalled();
if (_isVSCodeInstalled)
_context.PropertyChanged += Context_PropertyChanged;
}

public Task ExecuteAsync()
{
Win32API.RunPowershellCommand($"code {_context.ShellPage?.FilesystemViewModel.WorkingDirectory}", false);
return Task.CompletedTask;
}

private void Context_PropertyChanged(object? sender, PropertyChangedEventArgs e)
{
if (e.PropertyName == nameof(IContentPageContext.IsGitRepository))
OnPropertyChanged(nameof(IsExecutable));
}

private static bool IsVSCodeInstalled()
{
string registryKey = @"Software\Microsoft\Windows\CurrentVersion\Uninstall";

var key = Registry.CurrentUser.OpenSubKey(registryKey);
if (key is null)
return false;

string? displayName;

foreach (var subKey in key.GetSubKeyNames().Select(key.OpenSubKey))
{
displayName = subKey?.GetValue("DisplayName") as string;
if (!string.IsNullOrWhiteSpace(displayName) && displayName.StartsWith("Microsoft Visual Studio Code"))
{
key.Close();
return true;
ferrariofilippo marked this conversation as resolved.
Show resolved Hide resolved
}
}

key.Close();

return false;
}
}
}
2 changes: 2 additions & 0 deletions src/Files.App/Commands/CommandCodes.cs
Original file line number Diff line number Diff line change
Expand Up @@ -97,6 +97,8 @@ public enum CommandCodes
RotateRight,

// Open
OpenInVS,
OpenInVSCode,
OpenProperties,
OpenSettings,
OpenTerminal,
Expand Down
4 changes: 4 additions & 0 deletions src/Files.App/Commands/Manager/CommandManager.cs
Original file line number Diff line number Diff line change
Expand Up @@ -85,6 +85,8 @@ public IRichCommand this[HotKey hotKey]
public IRichCommand OpenItem => commands[CommandCodes.OpenItem];
public IRichCommand OpenItemWithApplicationPicker => commands[CommandCodes.OpenItemWithApplicationPicker];
public IRichCommand OpenParentFolder => commands[CommandCodes.OpenParentFolder];
public IRichCommand OpenInVS => commands[CommandCodes.OpenInVS];
public IRichCommand OpenInVSCode => commands[CommandCodes.OpenInVSCode];
public IRichCommand OpenProperties => commands[CommandCodes.OpenProperties];
public IRichCommand OpenSettings => commands[CommandCodes.OpenSettings];
public IRichCommand OpenTerminal => commands[CommandCodes.OpenTerminal];
Expand Down Expand Up @@ -236,6 +238,8 @@ public CommandManager()
[CommandCodes.OpenItem] = new OpenItemAction(),
[CommandCodes.OpenItemWithApplicationPicker] = new OpenItemWithApplicationPickerAction(),
[CommandCodes.OpenParentFolder] = new OpenParentFolderAction(),
[CommandCodes.OpenInVS] = new OpenInVSAction(),
[CommandCodes.OpenInVSCode] = new OpenInVSCodeAction(),
[CommandCodes.OpenProperties] = new OpenPropertiesAction(),
[CommandCodes.OpenSettings] = new OpenSettingsAction(),
[CommandCodes.OpenTerminal] = new OpenTerminalAction(),
Expand Down
2 changes: 2 additions & 0 deletions src/Files.App/Commands/Manager/ICommandManager.cs
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,8 @@ public interface ICommandManager : IEnumerable<IRichCommand>
IRichCommand RotateLeft { get; }
IRichCommand RotateRight { get; }

IRichCommand OpenInVS { get; }
IRichCommand OpenInVSCode { get; }
IRichCommand OpenProperties { get; }
IRichCommand OpenSettings { get; }
IRichCommand OpenTerminal { get; }
Expand Down
19 changes: 16 additions & 3 deletions src/Files.App/Contexts/ContentPage/ContentPageContext.cs
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,11 @@ internal class ContentPageContext : ObservableObject, IContentPageContext

public bool ShowSearchUnindexedItemsMessage => ShellPage is not null && ShellPage.InstanceViewModel.ShowSearchUnindexedItemsMessage;

public bool CanExecuteGitAction => ShellPage is not null && ShellPage.InstanceViewModel.IsGitRepository && !GitHelpers.IsExecutingGitAction;
public bool IsGitRepository => ShellPage is not null && ShellPage.InstanceViewModel.IsGitRepository;

public bool CanExecuteGitAction => IsGitRepository && !GitHelpers.IsExecutingGitAction;

public string? SolutionFilePath => ShellPage?.FilesystemViewModel.SolutionFilePath;

public ContentPageContext()
{
Expand Down Expand Up @@ -150,6 +154,7 @@ private void InstanceViewModel_PropertyChanged(object? sender, PropertyChangedEv
OnPropertyChanged(nameof(ShowSearchUnindexedItemsMessage));
break;
case nameof(CurrentInstanceViewModel.IsGitRepository):
OnPropertyChanged(nameof(IsGitRepository));
OnPropertyChanged(nameof(CanExecuteGitAction));
break;
}
Expand All @@ -175,8 +180,15 @@ private void ToolbarViewModel_PropertyChanged(object? sender, PropertyChangedEve

private void FilesystemViewModel_PropertyChanged(object? sender, PropertyChangedEventArgs e)
{
if (e.PropertyName is nameof(ItemViewModel.CurrentFolder))
OnPropertyChanged(nameof(Folder));
switch (e.PropertyName)
{
case nameof(ItemViewModel.CurrentFolder):
OnPropertyChanged(nameof(Folder));
break;
case nameof(ItemViewModel.SolutionFilePath):
OnPropertyChanged(nameof(SolutionFilePath));
break;
}
}

private void Update()
Expand All @@ -194,6 +206,7 @@ private void Update()
OnPropertyChanged(nameof(IsMultiPaneEnabled));
OnPropertyChanged(nameof(IsMultiPaneActive));
OnPropertyChanged(nameof(ShowSearchUnindexedItemsMessage));
OnPropertyChanged(nameof(IsGitRepository));
OnPropertyChanged(nameof(CanExecuteGitAction));
}

Expand Down
3 changes: 3 additions & 0 deletions src/Files.App/Contexts/ContentPage/IContentPageContext.cs
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,9 @@ public interface IContentPageContext : INotifyPropertyChanged

bool ShowSearchUnindexedItemsMessage { get; }

bool IsGitRepository { get; }
yaira2 marked this conversation as resolved.
Show resolved Hide resolved
bool CanExecuteGitAction { get; }

string? SolutionFilePath { get; }
}
}
24 changes: 23 additions & 1 deletion src/Files.App/Data/Models/ItemViewModel.cs
Original file line number Diff line number Diff line change
Expand Up @@ -8,8 +8,8 @@
using Files.App.Helpers.FileListCache;
using Files.App.Shell;
using Files.App.Storage.FtpStorage;
using Files.App.UserControls;
using Files.App.ViewModels.Previews;
using Files.Backend.Helpers;
using Files.Backend.Services;
using Files.Backend.Services.SizeProvider;
using Files.Backend.ViewModels.Dialogs;
Expand Down Expand Up @@ -76,6 +76,13 @@ public ListedItem? CurrentFolder
private set => SetProperty(ref currentFolder, value);
}

private string? solutionFilePath;
public string? SolutionFilePath
{
get => solutionFilePath;
private set => SetProperty(ref solutionFilePath, value);
}
ferrariofilippo marked this conversation as resolved.
Show resolved Hide resolved

public CollectionViewSource viewSource;

private FileSystemWatcher watcher;
Expand Down Expand Up @@ -1726,6 +1733,8 @@ await Task.Run(async () =>
}, defaultIconPairs: DefaultIcons);

filesAndFolders.AddRange(fileList);

await dispatcherQueue.EnqueueOrInvokeAsync(CheckForSolutionFile, Microsoft.UI.Dispatching.DispatcherQueuePriority.Low);
await OrderFilesAndFoldersAsync();
await ApplyFilesAndFoldersChangesAsync();
});
Expand Down Expand Up @@ -1763,6 +1772,19 @@ private Task EnumFromStorageFolderAsync(string path, BaseStorageFolder? rootFold
}, cancellationToken);
}

private void CheckForSolutionFile()
{
for (int i = 0; i < filesAndFolders.Count; i++)
{
if (FileExtensionHelpers.HasExtension(filesAndFolders[i].FileExtension, ".sln"))
{
SolutionFilePath = filesAndFolders[i].ItemPath;
return;
}
}
SolutionFilePath = null;
ferrariofilippo marked this conversation as resolved.
Show resolved Hide resolved
}

private async Task<CloudDriveSyncStatus> CheckCloudDriveSyncStatusAsync(IStorageItem item)
{
int? syncStatus = null;
Expand Down
12 changes: 12 additions & 0 deletions src/Files.App/Strings/en-US/Resources.resw
Original file line number Diff line number Diff line change
Expand Up @@ -3314,4 +3314,16 @@
<data name="ManageBranches" xml:space="preserve">
<value>Manage branches</value>
</data>
<data name="OpenInVS" xml:space="preserve">
<value>Open in VS</value>
ferrariofilippo marked this conversation as resolved.
Show resolved Hide resolved
</data>
<data name="OpenInVSDescription" xml:space="preserve">
<value>Open the current directory in Visual Studio</value>
</data>
<data name="OpenInVSCode" xml:space="preserve">
<value>Open in VS Code</value>
yaira2 marked this conversation as resolved.
Show resolved Hide resolved
</data>
<data name="OpenInVSCodeDescription" xml:space="preserve">
<value>Open the current directory in Visual Studio Code</value>
</data>
</root>
31 changes: 30 additions & 1 deletion src/Files.App/UserControls/StatusBarControl.xaml
Original file line number Diff line number Diff line change
Expand Up @@ -104,6 +104,7 @@
<Grid.ColumnDefinitions>
<ColumnDefinition />
<ColumnDefinition Width="Auto" />
<ColumnDefinition Width="Auto" />
</Grid.ColumnDefinitions>
<StackPanel
Grid.Column="0"
Expand Down Expand Up @@ -137,9 +138,37 @@
Visibility="{x:Bind SelectedItemsPropertiesViewModel.ItemSizeVisibility, Mode=OneWay}" />
</StackPanel>

<!-- Open in VS Code Button -->
<Button
x:Name="OpenInVSCodeButton"
Grid.Column="1"
Height="24"
Padding="8,0,8,0"
VerticalAlignment="Center"
x:Load="{x:Bind converters:MultiBooleanConverter.AndNotConvert(Commands.OpenInVSCode.IsExecutable, Commands.OpenInVS.IsExecutable), Mode=OneWay}"
Background="Transparent"
BorderBrush="Transparent"
Command="{x:Bind Commands.OpenInVSCode}"
Content="{x:Bind Commands.OpenInVSCode.Label}"
ToolTipService.ToolTip="{x:Bind Commands.OpenInVSCode.LabelWithHotKey, Mode=OneWay}" />

<!-- Open in VS Button -->
<Button
x:Name="OpenInVSButton"
Grid.Column="1"
Height="24"
Padding="8,0,8,0"
VerticalAlignment="Center"
x:Load="{x:Bind Commands.OpenInVS.IsExecutable, Mode=OneWay}"
Background="Transparent"
BorderBrush="Transparent"
Command="{x:Bind Commands.OpenInVS}"
Content="{x:Bind Commands.OpenInVS.Label}"
ToolTipService.ToolTip="{x:Bind Commands.OpenInVS.LabelWithHotKey, Mode=OneWay}" />
yaira2 marked this conversation as resolved.
Show resolved Hide resolved
yaira2 marked this conversation as resolved.
Show resolved Hide resolved

<StackPanel
x:Name="GitActionsPanel"
Grid.Column="1"
Grid.Column="2"
VerticalAlignment="Center"
x:Load="{x:Bind DirectoryPropertiesViewModel.GitBranchDisplayName, Mode=OneWay, Converter={StaticResource NullToFalseConverter}}"
Orientation="Horizontal"
Expand Down