600 lines
24 KiB
C++
600 lines
24 KiB
C++
// Copyright (c) Microsoft Corporation.
|
|
// Licensed under the MIT license.
|
|
|
|
#include "pch.h"
|
|
#include "Profiles.h"
|
|
|
|
#include "PreviewConnection.h"
|
|
#include "Profiles.g.cpp"
|
|
#include "EnumEntry.h"
|
|
|
|
#include <LibraryResources.h>
|
|
#include "..\WinRTUtils\inc\Utils.h"
|
|
|
|
// This function is a copy of DxFontInfo::_NearbyCollection() with
|
|
// * the call to DxFontInfo::s_GetNearbyFonts() inlined
|
|
// * checkForUpdates for GetSystemFontCollection() set to true
|
|
static wil::com_ptr<IDWriteFontCollection1> NearbyCollection(IDWriteFactory* dwriteFactory)
|
|
{
|
|
// The convenience interfaces for loading fonts from files
|
|
// are only available on Windows 10+.
|
|
wil::com_ptr<IDWriteFactory6> factory6;
|
|
// wil's query() facilities don't work inside WinRT land at the moment.
|
|
// They produce a compilation error due to IUnknown and winrt::Windows::Foundation::IUnknown being ambiguous.
|
|
if (!SUCCEEDED(dwriteFactory->QueryInterface(__uuidof(IDWriteFactory6), factory6.put_void())))
|
|
{
|
|
return nullptr;
|
|
}
|
|
|
|
wil::com_ptr<IDWriteFontCollection1> systemFontCollection;
|
|
THROW_IF_FAILED(factory6->GetSystemFontCollection(false, systemFontCollection.addressof(), true));
|
|
|
|
wil::com_ptr<IDWriteFontSet> systemFontSet;
|
|
THROW_IF_FAILED(systemFontCollection->GetFontSet(systemFontSet.addressof()));
|
|
|
|
wil::com_ptr<IDWriteFontSetBuilder2> fontSetBuilder2;
|
|
THROW_IF_FAILED(factory6->CreateFontSetBuilder(fontSetBuilder2.addressof()));
|
|
|
|
THROW_IF_FAILED(fontSetBuilder2->AddFontSet(systemFontSet.get()));
|
|
|
|
{
|
|
const std::filesystem::path module{ wil::GetModuleFileNameW<std::wstring>(nullptr) };
|
|
const auto folder{ module.parent_path() };
|
|
|
|
for (const auto& p : std::filesystem::directory_iterator(folder))
|
|
{
|
|
if (til::ends_with(p.path().native(), L".ttf"))
|
|
{
|
|
fontSetBuilder2->AddFontFile(p.path().c_str());
|
|
}
|
|
}
|
|
}
|
|
|
|
wil::com_ptr<IDWriteFontSet> fontSet;
|
|
THROW_IF_FAILED(fontSetBuilder2->CreateFontSet(fontSet.addressof()));
|
|
|
|
wil::com_ptr<IDWriteFontCollection1> fontCollection;
|
|
THROW_IF_FAILED(factory6->CreateFontCollectionFromFontSet(fontSet.get(), &fontCollection));
|
|
|
|
return fontCollection;
|
|
}
|
|
|
|
using namespace winrt::Windows::UI::Text;
|
|
using namespace winrt::Windows::UI::Xaml;
|
|
using namespace winrt::Windows::UI::Xaml::Controls;
|
|
using namespace winrt::Windows::UI::Xaml::Data;
|
|
using namespace winrt::Windows::UI::Xaml::Navigation;
|
|
using namespace winrt::Windows::Foundation;
|
|
using namespace winrt::Windows::Foundation::Collections;
|
|
using namespace winrt::Microsoft::Terminal::Settings::Model;
|
|
|
|
namespace winrt::Microsoft::Terminal::Settings::Editor::implementation
|
|
{
|
|
Windows::Foundation::Collections::IObservableVector<Editor::Font> ProfileViewModel::_MonospaceFontList{ nullptr };
|
|
Windows::Foundation::Collections::IObservableVector<Editor::Font> ProfileViewModel::_FontList{ nullptr };
|
|
|
|
ProfileViewModel::ProfileViewModel(const Model::Profile& profile, const Model::CascadiaSettings& appSettings) :
|
|
_profile{ profile },
|
|
_defaultAppearanceViewModel{ winrt::make<implementation::AppearanceViewModel>(profile.DefaultAppearance().try_as<AppearanceConfig>()) },
|
|
_originalProfileGuid{ profile.Guid() },
|
|
_appSettings{ appSettings },
|
|
_unfocusedAppearanceViewModel{ nullptr }
|
|
{
|
|
// Add a property changed handler to our own property changed event.
|
|
// This propagates changes from the settings model to anybody listening to our
|
|
// unique view model members.
|
|
PropertyChanged([this](auto&&, const PropertyChangedEventArgs& args) {
|
|
const auto viewModelProperty{ args.PropertyName() };
|
|
if (viewModelProperty == L"IsBaseLayer")
|
|
{
|
|
// we _always_ want to show the background image settings in base layer
|
|
_NotifyChanges(L"BackgroundImageSettingsVisible");
|
|
}
|
|
else if (viewModelProperty == L"StartingDirectory")
|
|
{
|
|
// notify listener that all starting directory related values might have changed
|
|
// NOTE: this is similar to what is done with BackgroundImagePath above
|
|
_NotifyChanges(L"UseParentProcessDirectory", L"UseCustomStartingDirectory");
|
|
}
|
|
else if (viewModelProperty == L"UseAcrylic")
|
|
{
|
|
// GH#11372: If we're on Windows 10, and someone turns off
|
|
// acrylic, we're going to disable opacity for them. Opacity
|
|
// doesn't work without acrylic on Windows 10.
|
|
//
|
|
// BODGY: CascadiaSettings's function IsDefaultTerminalAvailable
|
|
// is basically a "are we on Windows 11" check, because defterm
|
|
// only works on Win11. So we'll use that.
|
|
//
|
|
// Remove when we can remove the rest of GH#11285
|
|
if (!UseAcrylic() && !CascadiaSettings::IsDefaultTerminalAvailable())
|
|
{
|
|
Opacity(1.0);
|
|
}
|
|
}
|
|
});
|
|
|
|
// Do the same for the starting directory
|
|
if (!StartingDirectory().empty())
|
|
{
|
|
_lastStartingDirectoryPath = StartingDirectory();
|
|
}
|
|
|
|
// generate the font list, if we don't have one
|
|
if (!_FontList || !_MonospaceFontList)
|
|
{
|
|
UpdateFontList();
|
|
}
|
|
|
|
if (profile.HasUnfocusedAppearance())
|
|
{
|
|
_unfocusedAppearanceViewModel = winrt::make<implementation::AppearanceViewModel>(profile.UnfocusedAppearance().try_as<AppearanceConfig>());
|
|
}
|
|
|
|
_defaultAppearanceViewModel.IsDefault(true);
|
|
}
|
|
|
|
Model::TerminalSettings ProfileViewModel::TermSettings() const
|
|
{
|
|
return Model::TerminalSettings::CreateWithProfile(_appSettings, _profile, nullptr).DefaultSettings();
|
|
}
|
|
|
|
// Method Description:
|
|
// - Updates the lists of fonts and sorts them alphabetically
|
|
void ProfileViewModel::UpdateFontList() noexcept
|
|
try
|
|
{
|
|
// initialize font list
|
|
std::vector<Editor::Font> fontList;
|
|
std::vector<Editor::Font> monospaceFontList;
|
|
|
|
// get a DWriteFactory
|
|
com_ptr<IDWriteFactory> factory;
|
|
THROW_IF_FAILED(DWriteCreateFactory(
|
|
DWRITE_FACTORY_TYPE_SHARED,
|
|
__uuidof(IDWriteFactory),
|
|
reinterpret_cast<::IUnknown**>(factory.put())));
|
|
|
|
// get the font collection; subscribe to updates
|
|
const auto fontCollection = NearbyCollection(factory.get());
|
|
|
|
for (UINT32 i = 0; i < fontCollection->GetFontFamilyCount(); ++i)
|
|
{
|
|
try
|
|
{
|
|
// get the font family
|
|
com_ptr<IDWriteFontFamily> fontFamily;
|
|
THROW_IF_FAILED(fontCollection->GetFontFamily(i, fontFamily.put()));
|
|
|
|
// get the font's localized names
|
|
com_ptr<IDWriteLocalizedStrings> localizedFamilyNames;
|
|
THROW_IF_FAILED(fontFamily->GetFamilyNames(localizedFamilyNames.put()));
|
|
|
|
// construct a font entry for tracking
|
|
if (auto fontEntry{ _GetFont(localizedFamilyNames) })
|
|
{
|
|
// check if the font is monospaced
|
|
try
|
|
{
|
|
com_ptr<IDWriteFont> font;
|
|
THROW_IF_FAILED(fontFamily->GetFirstMatchingFont(DWRITE_FONT_WEIGHT::DWRITE_FONT_WEIGHT_NORMAL,
|
|
DWRITE_FONT_STRETCH::DWRITE_FONT_STRETCH_NORMAL,
|
|
DWRITE_FONT_STYLE::DWRITE_FONT_STYLE_NORMAL,
|
|
font.put()));
|
|
|
|
// add the font name to our list of monospace fonts
|
|
const auto castedFont{ font.try_as<IDWriteFont1>() };
|
|
if (castedFont && castedFont->IsMonospacedFont())
|
|
{
|
|
monospaceFontList.emplace_back(fontEntry);
|
|
}
|
|
}
|
|
CATCH_LOG();
|
|
|
|
// add the font name to our list of all fonts
|
|
fontList.emplace_back(std::move(fontEntry));
|
|
}
|
|
}
|
|
CATCH_LOG();
|
|
}
|
|
|
|
// sort and save the lists
|
|
std::sort(begin(fontList), end(fontList), FontComparator());
|
|
_FontList = single_threaded_observable_vector<Editor::Font>(std::move(fontList));
|
|
|
|
std::sort(begin(monospaceFontList), end(monospaceFontList), FontComparator());
|
|
_MonospaceFontList = single_threaded_observable_vector<Editor::Font>(std::move(monospaceFontList));
|
|
}
|
|
CATCH_LOG();
|
|
|
|
Editor::Font ProfileViewModel::_GetFont(com_ptr<IDWriteLocalizedStrings> localizedFamilyNames)
|
|
{
|
|
// used for the font's name as an identifier (i.e. text block's font family property)
|
|
std::wstring nameID;
|
|
UINT32 nameIDIndex;
|
|
|
|
// used for the font's localized name
|
|
std::wstring localizedName;
|
|
UINT32 localizedNameIndex;
|
|
|
|
// use our current locale to find the localized name
|
|
BOOL exists{ FALSE };
|
|
HRESULT hr;
|
|
wchar_t localeName[LOCALE_NAME_MAX_LENGTH];
|
|
if (GetUserDefaultLocaleName(localeName, LOCALE_NAME_MAX_LENGTH))
|
|
{
|
|
hr = localizedFamilyNames->FindLocaleName(localeName, &localizedNameIndex, &exists);
|
|
}
|
|
if (SUCCEEDED(hr) && !exists)
|
|
{
|
|
// if we can't find the font for our locale, fallback to the en-us one
|
|
// Source: https://docs.microsoft.com/en-us/windows/win32/api/dwrite/nf-dwrite-idwritelocalizedstrings-findlocalename
|
|
hr = localizedFamilyNames->FindLocaleName(L"en-us", &localizedNameIndex, &exists);
|
|
}
|
|
if (!exists)
|
|
{
|
|
// failed to find the correct locale, using the first one
|
|
localizedNameIndex = 0;
|
|
}
|
|
|
|
// get the localized name
|
|
UINT32 nameLength;
|
|
THROW_IF_FAILED(localizedFamilyNames->GetStringLength(localizedNameIndex, &nameLength));
|
|
|
|
localizedName.resize(nameLength);
|
|
THROW_IF_FAILED(localizedFamilyNames->GetString(localizedNameIndex, localizedName.data(), nameLength + 1));
|
|
|
|
// now get the nameID
|
|
hr = localizedFamilyNames->FindLocaleName(L"en-us", &nameIDIndex, &exists);
|
|
if (FAILED(hr) || !exists)
|
|
{
|
|
// failed to find it, using the first one
|
|
nameIDIndex = 0;
|
|
}
|
|
|
|
// get the nameID
|
|
THROW_IF_FAILED(localizedFamilyNames->GetStringLength(nameIDIndex, &nameLength));
|
|
nameID.resize(nameLength);
|
|
THROW_IF_FAILED(localizedFamilyNames->GetString(nameIDIndex, nameID.data(), nameLength + 1));
|
|
|
|
if (!nameID.empty() && !localizedName.empty())
|
|
{
|
|
return make<Font>(nameID, localizedName);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
IObservableVector<Editor::Font> ProfileViewModel::CompleteFontList() const noexcept
|
|
{
|
|
return _FontList;
|
|
}
|
|
|
|
IObservableVector<Editor::Font> ProfileViewModel::MonospaceFontList() const noexcept
|
|
{
|
|
return _MonospaceFontList;
|
|
}
|
|
|
|
winrt::guid ProfileViewModel::OriginalProfileGuid() const noexcept
|
|
{
|
|
return _originalProfileGuid;
|
|
}
|
|
|
|
bool ProfileViewModel::CanDeleteProfile() const
|
|
{
|
|
return !IsBaseLayer();
|
|
}
|
|
|
|
Editor::AppearanceViewModel ProfileViewModel::DefaultAppearance()
|
|
{
|
|
return _defaultAppearanceViewModel;
|
|
}
|
|
|
|
bool ProfileViewModel::HasUnfocusedAppearance()
|
|
{
|
|
return _profile.HasUnfocusedAppearance();
|
|
}
|
|
|
|
bool ProfileViewModel::EditableUnfocusedAppearance() const noexcept
|
|
{
|
|
return Feature_EditableUnfocusedAppearance::IsEnabled();
|
|
}
|
|
|
|
bool ProfileViewModel::ShowUnfocusedAppearance()
|
|
{
|
|
return EditableUnfocusedAppearance() && HasUnfocusedAppearance();
|
|
}
|
|
|
|
void ProfileViewModel::CreateUnfocusedAppearance(const Windows::Foundation::Collections::IMapView<hstring, Model::ColorScheme>& schemes,
|
|
const IHostedInWindow& windowRoot)
|
|
{
|
|
_profile.CreateUnfocusedAppearance();
|
|
|
|
_unfocusedAppearanceViewModel = winrt::make<implementation::AppearanceViewModel>(_profile.UnfocusedAppearance().try_as<AppearanceConfig>());
|
|
_unfocusedAppearanceViewModel.Schemes(schemes);
|
|
_unfocusedAppearanceViewModel.WindowRoot(windowRoot);
|
|
|
|
_NotifyChanges(L"UnfocusedAppearance", L"HasUnfocusedAppearance", L"ShowUnfocusedAppearance");
|
|
}
|
|
|
|
void ProfileViewModel::DeleteUnfocusedAppearance()
|
|
{
|
|
_profile.DeleteUnfocusedAppearance();
|
|
|
|
_unfocusedAppearanceViewModel = nullptr;
|
|
|
|
_NotifyChanges(L"UnfocusedAppearance", L"HasUnfocusedAppearance", L"ShowUnfocusedAppearance");
|
|
}
|
|
|
|
Editor::AppearanceViewModel ProfileViewModel::UnfocusedAppearance()
|
|
{
|
|
return _unfocusedAppearanceViewModel;
|
|
}
|
|
|
|
bool ProfileViewModel::AtlasEngineAvailable() const noexcept
|
|
{
|
|
return Feature_AtlasEngine::IsEnabled();
|
|
}
|
|
|
|
bool ProfileViewModel::UseParentProcessDirectory()
|
|
{
|
|
return StartingDirectory().empty();
|
|
}
|
|
|
|
// This function simply returns the opposite of UseParentProcessDirectory.
|
|
// We bind the 'IsEnabled' parameters of the textbox and browse button
|
|
// to this because it needs to be the reverse of UseParentProcessDirectory
|
|
// but we don't want to create a whole new converter for inverting a boolean
|
|
bool ProfileViewModel::UseCustomStartingDirectory()
|
|
{
|
|
return !UseParentProcessDirectory();
|
|
}
|
|
|
|
void ProfileViewModel::UseParentProcessDirectory(const bool useParent)
|
|
{
|
|
if (useParent)
|
|
{
|
|
// Stash the current value of StartingDirectory. If the user
|
|
// checks and un-checks the "Use parent process directory" button, we want
|
|
// the path that we display in the text box to remain unchanged.
|
|
//
|
|
// Only stash this value if it's not empty
|
|
if (!StartingDirectory().empty())
|
|
{
|
|
_lastStartingDirectoryPath = StartingDirectory();
|
|
}
|
|
StartingDirectory(L"");
|
|
}
|
|
else
|
|
{
|
|
// Restore the path we had previously cached as long as it wasn't empty
|
|
// If it was empty, set the starting directory to %USERPROFILE%
|
|
// (we need to set it to something non-empty otherwise we will automatically
|
|
// disable the text box)
|
|
if (_lastStartingDirectoryPath.empty())
|
|
{
|
|
StartingDirectory(L"%USERPROFILE%");
|
|
}
|
|
else
|
|
{
|
|
StartingDirectory(_lastStartingDirectoryPath);
|
|
}
|
|
}
|
|
}
|
|
|
|
void ProfilePageNavigationState::DeleteProfile()
|
|
{
|
|
auto deleteProfileArgs{ winrt::make_self<DeleteProfileEventArgs>(_Profile.Guid()) };
|
|
_DeleteProfileHandlers(*this, *deleteProfileArgs);
|
|
}
|
|
|
|
void ProfilePageNavigationState::CreateUnfocusedAppearance()
|
|
{
|
|
_Profile.CreateUnfocusedAppearance(_Schemes, _WindowRoot);
|
|
}
|
|
|
|
void ProfilePageNavigationState::DeleteUnfocusedAppearance()
|
|
{
|
|
_Profile.DeleteUnfocusedAppearance();
|
|
}
|
|
|
|
Profiles::Profiles() :
|
|
_previewControl{ Control::TermControl(Model::TerminalSettings{}, make<PreviewConnection>()) }
|
|
{
|
|
InitializeComponent();
|
|
|
|
INITIALIZE_BINDABLE_ENUM_SETTING(AntiAliasingMode, TextAntialiasingMode, winrt::Microsoft::Terminal::Control::TextAntialiasingMode, L"Profile_AntialiasingMode", L"Content");
|
|
INITIALIZE_BINDABLE_ENUM_SETTING_REVERSE_ORDER(CloseOnExitMode, CloseOnExitMode, winrt::Microsoft::Terminal::Settings::Model::CloseOnExitMode, L"Profile_CloseOnExit", L"Content");
|
|
INITIALIZE_BINDABLE_ENUM_SETTING(ScrollState, ScrollbarState, winrt::Microsoft::Terminal::Control::ScrollbarState, L"Profile_ScrollbarVisibility", L"Content");
|
|
|
|
const auto startingDirCheckboxTooltip{ ToolTipService::GetToolTip(StartingDirectoryUseParentCheckbox()) };
|
|
Automation::AutomationProperties::SetFullDescription(StartingDirectoryUseParentCheckbox(), unbox_value<hstring>(startingDirCheckboxTooltip));
|
|
|
|
Automation::AutomationProperties::SetName(DeleteButton(), RS_(L"Profile_DeleteButton/Text"));
|
|
|
|
_previewControl.IsEnabled(false);
|
|
_previewControl.AllowFocusWhenDisabled(false);
|
|
ControlPreview().Child(_previewControl);
|
|
}
|
|
|
|
void Profiles::OnNavigatedTo(const NavigationEventArgs& e)
|
|
{
|
|
_State = e.Parameter().as<Editor::ProfilePageNavigationState>();
|
|
|
|
// generate the font list, if we don't have one
|
|
if (!_State.Profile().CompleteFontList() || !_State.Profile().MonospaceFontList())
|
|
{
|
|
ProfileViewModel::UpdateFontList();
|
|
}
|
|
|
|
// Check the use parent directory box if the starting directory is empty
|
|
if (_State.Profile().StartingDirectory().empty())
|
|
{
|
|
StartingDirectoryUseParentCheckbox().IsChecked(true);
|
|
}
|
|
|
|
// Subscribe to some changes in the view model
|
|
// These changes should force us to update our own set of "Current<Setting>" members,
|
|
// and propagate those changes to the UI
|
|
_ViewModelChangedRevoker = _State.Profile().PropertyChanged(winrt::auto_revoke, [=](auto&&, const PropertyChangedEventArgs& args) {
|
|
const auto settingName{ args.PropertyName() };
|
|
if (settingName == L"AntialiasingMode")
|
|
{
|
|
_PropertyChangedHandlers(*this, PropertyChangedEventArgs{ L"CurrentAntiAliasingMode" });
|
|
}
|
|
else if (settingName == L"CloseOnExit")
|
|
{
|
|
_PropertyChangedHandlers(*this, PropertyChangedEventArgs{ L"CurrentCloseOnExitMode" });
|
|
}
|
|
else if (settingName == L"BellStyle")
|
|
{
|
|
_PropertyChangedHandlers(*this, PropertyChangedEventArgs{ L"IsBellStyleFlagSet" });
|
|
}
|
|
else if (settingName == L"ScrollState")
|
|
{
|
|
_PropertyChangedHandlers(*this, PropertyChangedEventArgs{ L"CurrentScrollState" });
|
|
}
|
|
_previewControl.Settings(_State.Profile().TermSettings());
|
|
_previewControl.UpdateSettings();
|
|
});
|
|
|
|
// The Appearances object handles updating the values in the settings UI, but
|
|
// we still need to listen to the changes here just to update the preview control
|
|
_AppearanceViewModelChangedRevoker = _State.Profile().DefaultAppearance().PropertyChanged(winrt::auto_revoke, [=](auto&&, const PropertyChangedEventArgs& /*args*/) {
|
|
_previewControl.Settings(_State.Profile().TermSettings());
|
|
_previewControl.UpdateSettings();
|
|
});
|
|
|
|
// Navigate to the pivot in the provided navigation state
|
|
ProfilesPivot().SelectedIndex(static_cast<int>(_State.LastActivePivot()));
|
|
|
|
_previewControl.Settings(_State.Profile().TermSettings());
|
|
// There is a possibility that the control has not fully initialized yet,
|
|
// so wait for it to initialize before updating the settings (so we know
|
|
// that the renderer is set up)
|
|
_previewControl.Initialized([&](auto&& /*s*/, auto&& /*e*/) {
|
|
_previewControl.UpdateSettings();
|
|
});
|
|
}
|
|
|
|
void Profiles::OnNavigatedFrom(const NavigationEventArgs& /*e*/)
|
|
{
|
|
_ViewModelChangedRevoker.revoke();
|
|
_AppearanceViewModelChangedRevoker.revoke();
|
|
}
|
|
|
|
bool Profiles::IsBellStyleFlagSet(const uint32_t flag)
|
|
{
|
|
return (WI_EnumValue(_State.Profile().BellStyle()) & flag) == flag;
|
|
}
|
|
|
|
void Profiles::SetBellStyleAudible(winrt::Windows::Foundation::IReference<bool> on)
|
|
{
|
|
auto currentStyle = State().Profile().BellStyle();
|
|
WI_UpdateFlag(currentStyle, Model::BellStyle::Audible, winrt::unbox_value<bool>(on));
|
|
State().Profile().BellStyle(currentStyle);
|
|
}
|
|
|
|
void Profiles::SetBellStyleWindow(winrt::Windows::Foundation::IReference<bool> on)
|
|
{
|
|
auto currentStyle = State().Profile().BellStyle();
|
|
WI_UpdateFlag(currentStyle, Model::BellStyle::Window, winrt::unbox_value<bool>(on));
|
|
State().Profile().BellStyle(currentStyle);
|
|
}
|
|
|
|
void Profiles::SetBellStyleTaskbar(winrt::Windows::Foundation::IReference<bool> on)
|
|
{
|
|
auto currentStyle = State().Profile().BellStyle();
|
|
WI_UpdateFlag(currentStyle, Model::BellStyle::Taskbar, winrt::unbox_value<bool>(on));
|
|
State().Profile().BellStyle(currentStyle);
|
|
}
|
|
|
|
void Profiles::DeleteConfirmation_Click(IInspectable const& /*sender*/, RoutedEventArgs const& /*e*/)
|
|
{
|
|
auto state{ winrt::get_self<ProfilePageNavigationState>(_State) };
|
|
state->DeleteProfile();
|
|
}
|
|
|
|
void Profiles::CreateUnfocusedAppearance_Click(IInspectable const& /*sender*/, RoutedEventArgs const& /*e*/)
|
|
{
|
|
_State.CreateUnfocusedAppearance();
|
|
}
|
|
|
|
void Profiles::DeleteUnfocusedAppearance_Click(IInspectable const& /*sender*/, RoutedEventArgs const& /*e*/)
|
|
{
|
|
_State.DeleteUnfocusedAppearance();
|
|
}
|
|
|
|
fire_and_forget Profiles::Icon_Click(IInspectable const&, RoutedEventArgs const&)
|
|
{
|
|
auto lifetime = get_strong();
|
|
|
|
const auto parentHwnd{ reinterpret_cast<HWND>(_State.WindowRoot().GetHostingWindow()) };
|
|
auto file = co_await OpenImagePicker(parentHwnd);
|
|
if (!file.empty())
|
|
{
|
|
_State.Profile().Icon(file);
|
|
}
|
|
}
|
|
|
|
fire_and_forget Profiles::Commandline_Click(IInspectable const&, RoutedEventArgs const&)
|
|
{
|
|
auto lifetime = get_strong();
|
|
|
|
static constexpr COMDLG_FILTERSPEC supportedFileTypes[] = {
|
|
{ L"Executable Files (*.exe, *.cmd, *.bat)", L"*.exe;*.cmd;*.bat" },
|
|
{ L"All Files (*.*)", L"*.*" }
|
|
};
|
|
|
|
static constexpr winrt::guid clientGuidExecutables{ 0x2E7E4331, 0x0800, 0x48E6, { 0xB0, 0x17, 0xA1, 0x4C, 0xD8, 0x73, 0xDD, 0x58 } };
|
|
const auto parentHwnd{ reinterpret_cast<HWND>(_State.WindowRoot().GetHostingWindow()) };
|
|
auto path = co_await OpenFilePicker(parentHwnd, [](auto&& dialog) {
|
|
THROW_IF_FAILED(dialog->SetClientGuid(clientGuidExecutables));
|
|
try
|
|
{
|
|
auto folderShellItem{ winrt::capture<IShellItem>(&SHGetKnownFolderItem, FOLDERID_ComputerFolder, KF_FLAG_DEFAULT, nullptr) };
|
|
dialog->SetDefaultFolder(folderShellItem.get());
|
|
}
|
|
CATCH_LOG(); // non-fatal
|
|
THROW_IF_FAILED(dialog->SetFileTypes(ARRAYSIZE(supportedFileTypes), supportedFileTypes));
|
|
THROW_IF_FAILED(dialog->SetFileTypeIndex(1)); // the array is 1-indexed
|
|
THROW_IF_FAILED(dialog->SetDefaultExtension(L"exe;cmd;bat"));
|
|
});
|
|
|
|
if (!path.empty())
|
|
{
|
|
_State.Profile().Commandline(path);
|
|
}
|
|
}
|
|
|
|
fire_and_forget Profiles::StartingDirectory_Click(IInspectable const&, RoutedEventArgs const&)
|
|
{
|
|
auto lifetime = get_strong();
|
|
const auto parentHwnd{ reinterpret_cast<HWND>(_State.WindowRoot().GetHostingWindow()) };
|
|
auto folder = co_await OpenFilePicker(parentHwnd, [](auto&& dialog) {
|
|
static constexpr winrt::guid clientGuidFolderPicker{ 0xAADAA433, 0xB04D, 0x4BAE, { 0xB1, 0xEA, 0x1E, 0x6C, 0xD1, 0xCD, 0xA6, 0x8B } };
|
|
THROW_IF_FAILED(dialog->SetClientGuid(clientGuidFolderPicker));
|
|
try
|
|
{
|
|
auto folderShellItem{ winrt::capture<IShellItem>(&SHGetKnownFolderItem, FOLDERID_ComputerFolder, KF_FLAG_DEFAULT, nullptr) };
|
|
dialog->SetDefaultFolder(folderShellItem.get());
|
|
}
|
|
CATCH_LOG(); // non-fatal
|
|
|
|
DWORD flags{};
|
|
THROW_IF_FAILED(dialog->GetOptions(&flags));
|
|
THROW_IF_FAILED(dialog->SetOptions(flags | FOS_PICKFOLDERS)); // folders only
|
|
});
|
|
|
|
if (!folder.empty())
|
|
{
|
|
_State.Profile().StartingDirectory(folder);
|
|
}
|
|
}
|
|
|
|
void Profiles::Pivot_SelectionChanged(Windows::Foundation::IInspectable const& /*sender*/,
|
|
RoutedEventArgs const& /*e*/)
|
|
{
|
|
_State.LastActivePivot(static_cast<Editor::ProfilesPivots>(ProfilesPivot().SelectedIndex()));
|
|
}
|
|
}
|