From d60517e55f26868c03a9d831804edced1c791e9c Mon Sep 17 00:00:00 2001
From: Ryan <69221034+ryfu-msft@users.noreply.github.com>
Date: Mon, 28 Oct 2024 17:31:05 -0400
Subject: [PATCH] Add experimental feature for font list command (#4886)
---
.github/actions/spelling/expect.txt | 1 +
.../JSON/settings/settings.schema.0.2.json | 5 +
.../AppInstallerCLICore.vcxproj | 4 +
.../AppInstallerCLICore.vcxproj.filters | 12 ++
src/AppInstallerCLICore/Argument.cpp | 6 +
.../Commands/FontCommand.cpp | 86 +++++++++
.../Commands/FontCommand.h | 40 ++++
.../Commands/RootCommand.cpp | 2 +
src/AppInstallerCLICore/ExecutionArgs.h | 3 +
src/AppInstallerCLICore/Resources.h | 11 ++
.../Workflows/FontFlow.cpp | 124 +++++++++++++
src/AppInstallerCLICore/Workflows/FontFlow.h | 13 ++
src/AppInstallerCLIE2ETests/BaseCommand.cs | 5 +-
.../Shared/Strings/en-us/winget.resw | 35 ++++
.../AppInstallerCLITests.vcxproj | 1 +
.../AppInstallerCLITests.vcxproj.filters | 3 +
src/AppInstallerCLITests/Fonts.cpp | 37 ++++
src/AppInstallerCLITests/Versions.cpp | 27 +++
.../AppInstallerCommonCore.vcxproj | 2 +
.../AppInstallerCommonCore.vcxproj.filters | 6 +
.../ExperimentalFeature.cpp | 5 +
src/AppInstallerCommonCore/Fonts.cpp | 175 ++++++++++++++++++
src/AppInstallerCommonCore/Locale.cpp | 12 ++
.../Public/AppInstallerRuntime.h | 7 +
.../Public/winget/ExperimentalFeature.h | 1 +
.../Public/winget/Fonts.h | 39 ++++
.../Public/winget/Locale.h | 5 +-
.../Public/winget/UserSettings.h | 2 +
src/AppInstallerCommonCore/Runtime.cpp | 31 +++-
src/AppInstallerCommonCore/UserSettings.cpp | 1 +
.../Public/AppInstallerVersions.h | 16 ++
src/AppInstallerSharedLib/Versions.cpp | 60 ++++++
32 files changed, 763 insertions(+), 14 deletions(-)
create mode 100644 src/AppInstallerCLICore/Commands/FontCommand.cpp
create mode 100644 src/AppInstallerCLICore/Commands/FontCommand.h
create mode 100644 src/AppInstallerCLICore/Workflows/FontFlow.cpp
create mode 100644 src/AppInstallerCLICore/Workflows/FontFlow.h
create mode 100644 src/AppInstallerCLITests/Fonts.cpp
create mode 100644 src/AppInstallerCommonCore/Fonts.cpp
create mode 100644 src/AppInstallerCommonCore/Public/winget/Fonts.h
diff --git a/.github/actions/spelling/expect.txt b/.github/actions/spelling/expect.txt
index f063332bb8..8c2cff836e 100644
--- a/.github/actions/spelling/expect.txt
+++ b/.github/actions/spelling/expect.txt
@@ -143,6 +143,7 @@ DUPLICATEALIAS
dustojnikhummer
dvinns
dwgs
+dwrite
ecfr
ecfrbrowse
EFGH
diff --git a/schemas/JSON/settings/settings.schema.0.2.json b/schemas/JSON/settings/settings.schema.0.2.json
index 7cc2001c1d..2113127f0e 100644
--- a/schemas/JSON/settings/settings.schema.0.2.json
+++ b/schemas/JSON/settings/settings.schema.0.2.json
@@ -287,6 +287,11 @@
"description": "Enable support for the configure export command",
"type": "boolean",
"default": false
+ },
+ "fonts": {
+ "description": "Enable support for managing fonts",
+ "type": "boolean",
+ "default": false
}
}
}
diff --git a/src/AppInstallerCLICore/AppInstallerCLICore.vcxproj b/src/AppInstallerCLICore/AppInstallerCLICore.vcxproj
index 58c83d1efb..45c9c0fa04 100644
--- a/src/AppInstallerCLICore/AppInstallerCLICore.vcxproj
+++ b/src/AppInstallerCLICore/AppInstallerCLICore.vcxproj
@@ -364,6 +364,7 @@
+
@@ -410,6 +411,7 @@
+
@@ -441,6 +443,7 @@
+
@@ -490,6 +493,7 @@
+
diff --git a/src/AppInstallerCLICore/AppInstallerCLICore.vcxproj.filters b/src/AppInstallerCLICore/AppInstallerCLICore.vcxproj.filters
index 6c072ed000..a8f48edd34 100644
--- a/src/AppInstallerCLICore/AppInstallerCLICore.vcxproj.filters
+++ b/src/AppInstallerCLICore/AppInstallerCLICore.vcxproj.filters
@@ -260,6 +260,12 @@
Header Files
+
+ Commands
+
+
+ Workflows
+
@@ -490,6 +496,12 @@
Source Files
+
+ Commands
+
+
+ Workflows
+
diff --git a/src/AppInstallerCLICore/Argument.cpp b/src/AppInstallerCLICore/Argument.cpp
index a5334a2ceb..0b1864df3c 100644
--- a/src/AppInstallerCLICore/Argument.cpp
+++ b/src/AppInstallerCLICore/Argument.cpp
@@ -198,6 +198,10 @@ namespace AppInstaller::CLI
case Execution::Args::Type::IgnoreResumeLimit:
return { type, "ignore-resume-limit"_liv, ArgTypeCategory::None };
+ // Font command
+ case Execution::Args::Type::Family:
+ return { type, "family"_liv, ArgTypeCategory::None };
+
// Configuration commands
case Execution::Args::Type::ConfigurationFile:
return { type, "file"_liv, 'f', ArgTypeCategory::ConfigurationSetChoice, ArgTypeExclusiveSet::ConfigurationSetChoice };
@@ -430,6 +434,8 @@ namespace AppInstaller::CLI
return Argument{ type, Resource::String::ProxyArgumentDescription, ArgumentType::Standard, TogglePolicy::Policy::ProxyCommandLineOptions, BoolAdminSetting::ProxyCommandLineOptions };
case Args::Type::NoProxy:
return Argument{ type, Resource::String::NoProxyArgumentDescription, ArgumentType::Flag, TogglePolicy::Policy::ProxyCommandLineOptions, BoolAdminSetting::ProxyCommandLineOptions };
+ case Args::Type::Family:
+ return Argument{ type, Resource::String::FontFamilyNameArgumentDescription, ArgumentType::Positional, false };
default:
THROW_HR(E_UNEXPECTED);
}
diff --git a/src/AppInstallerCLICore/Commands/FontCommand.cpp b/src/AppInstallerCLICore/Commands/FontCommand.cpp
new file mode 100644
index 0000000000..c6faae5b44
--- /dev/null
+++ b/src/AppInstallerCLICore/Commands/FontCommand.cpp
@@ -0,0 +1,86 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+#include "pch.h"
+#include "FontCommand.h"
+#include "Workflows/CompletionFlow.h"
+#include "Workflows/WorkflowBase.h"
+#include "Workflows/FontFlow.h"
+#include "Resources.h"
+
+namespace AppInstaller::CLI
+{
+ using namespace AppInstaller::CLI::Execution;
+ using namespace AppInstaller::CLI::Workflow;
+ using namespace AppInstaller::Utility::literals;
+ using namespace std::string_view_literals;
+
+ Utility::LocIndView s_FontCommand_HelpLink = "https://aka.ms/winget-command-font"_liv;
+
+ std::vector> FontCommand::GetCommands() const
+ {
+ return InitializeFromMoveOnly>>({
+ std::make_unique(FullName()),
+ });
+ }
+
+ Resource::LocString FontCommand::ShortDescription() const
+ {
+ return { Resource::String::FontCommandShortDescription };
+ }
+
+ Resource::LocString FontCommand::LongDescription() const
+ {
+ return { Resource::String::FontCommandLongDescription };
+ }
+
+ Utility::LocIndView FontCommand::HelpLink() const
+ {
+ return s_FontCommand_HelpLink;
+ }
+
+ void FontCommand::ExecuteInternal(Execution::Context& context) const
+ {
+ OutputHelp(context.Reporter);
+ }
+
+ std::vector FontListCommand::GetArguments() const
+ {
+ return {
+ Argument::ForType(Args::Type::Family),
+ Argument::ForType(Args::Type::Moniker),
+ Argument::ForType(Args::Type::Source),
+ Argument::ForType(Args::Type::Tag),
+ Argument::ForType(Args::Type::Exact),
+ Argument::ForType(Args::Type::AuthenticationMode),
+ Argument::ForType(Args::Type::AuthenticationAccount),
+ Argument::ForType(Args::Type::AcceptSourceAgreements),
+ };
+ }
+
+ Resource::LocString FontListCommand::ShortDescription() const
+ {
+ return { Resource::String::FontListCommandShortDescription };
+ }
+
+ Resource::LocString FontListCommand::LongDescription() const
+ {
+ return { Resource::String::FontListCommandLongDescription };
+ }
+
+ void FontListCommand::Complete(Execution::Context& context, Args::Type valueType) const
+ {
+ UNREFERENCED_PARAMETER(valueType);
+ context.Reporter.Error() << Resource::String::PendingWorkError << std::endl;
+ THROW_HR(E_NOTIMPL);
+ }
+
+ Utility::LocIndView FontListCommand::HelpLink() const
+ {
+ return s_FontCommand_HelpLink;
+ }
+
+ void FontListCommand::ExecuteInternal(Execution::Context& context) const
+ {
+ context << Workflow::ReportInstalledFonts;
+ }
+}
diff --git a/src/AppInstallerCLICore/Commands/FontCommand.h b/src/AppInstallerCLICore/Commands/FontCommand.h
new file mode 100644
index 0000000000..c29f703f6c
--- /dev/null
+++ b/src/AppInstallerCLICore/Commands/FontCommand.h
@@ -0,0 +1,40 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+#pragma once
+#include "Command.h"
+#include
+
+namespace AppInstaller::CLI
+{
+ struct FontCommand final : public Command
+ {
+ FontCommand(std::string_view parent) : Command("font", { "fonts" }, parent, Settings::ExperimentalFeature::Feature::Font) {}
+
+ std::vector> GetCommands() const override;
+
+ Resource::LocString ShortDescription() const override;
+ Resource::LocString LongDescription() const override;
+
+ Utility::LocIndView HelpLink() const override;
+
+ protected:
+ void ExecuteInternal(Execution::Context& context) const override;
+ };
+
+ struct FontListCommand final : public Command
+ {
+ FontListCommand(std::string_view parent) : Command("list", parent) {}
+
+ std::vector GetArguments() const override;
+
+ Resource::LocString ShortDescription() const override;
+ Resource::LocString LongDescription() const override;
+
+ void Complete(Execution::Context& context, Execution::Args::Type valueType) const override;
+
+ Utility::LocIndView HelpLink() const override;
+
+ protected:
+ void ExecuteInternal(Execution::Context& context) const override;
+ };
+}
diff --git a/src/AppInstallerCLICore/Commands/RootCommand.cpp b/src/AppInstallerCLICore/Commands/RootCommand.cpp
index f5a61d79fa..5580efa14a 100644
--- a/src/AppInstallerCLICore/Commands/RootCommand.cpp
+++ b/src/AppInstallerCLICore/Commands/RootCommand.cpp
@@ -15,6 +15,7 @@
#include "ValidateCommand.h"
#include "SettingsCommand.h"
#include "FeaturesCommand.h"
+#include "FontCommand.h"
#include "ExperimentalCommand.h"
#include "CompleteCommand.h"
#include "ExportCommand.h"
@@ -194,6 +195,7 @@ namespace AppInstaller::CLI
std::make_unique(FullName()),
std::make_unique(FullName()),
std::make_unique(FullName()),
+ std::make_unique(FullName()),
#if _DEBUG
std::make_unique(FullName()),
#endif
diff --git a/src/AppInstallerCLICore/ExecutionArgs.h b/src/AppInstallerCLICore/ExecutionArgs.h
index 41739b36f4..7bb2c28628 100644
--- a/src/AppInstallerCLICore/ExecutionArgs.h
+++ b/src/AppInstallerCLICore/ExecutionArgs.h
@@ -121,6 +121,9 @@ namespace AppInstaller::CLI::Execution
ResumeId,
IgnoreResumeLimit,
+ // Font Command
+ Family,
+
// Configuration
ConfigurationFile,
ConfigurationAcceptWarning,
diff --git a/src/AppInstallerCLICore/Resources.h b/src/AppInstallerCLICore/Resources.h
index 0e8407ab2e..5aa7616509 100644
--- a/src/AppInstallerCLICore/Resources.h
+++ b/src/AppInstallerCLICore/Resources.h
@@ -248,6 +248,16 @@ namespace AppInstaller::CLI::Resource
WINGET_DEFINE_RESOURCE_STRINGID(FileNotFound);
WINGET_DEFINE_RESOURCE_STRINGID(FilesRemainInInstallDirectory);
WINGET_DEFINE_RESOURCE_STRINGID(FlagContainAdjoinedError);
+ WINGET_DEFINE_RESOURCE_STRINGID(FontCommandLongDescription);
+ WINGET_DEFINE_RESOURCE_STRINGID(FontCommandShortDescription);
+ WINGET_DEFINE_RESOURCE_STRINGID(FontFace);
+ WINGET_DEFINE_RESOURCE_STRINGID(FontFaces);
+ WINGET_DEFINE_RESOURCE_STRINGID(FontFamily);
+ WINGET_DEFINE_RESOURCE_STRINGID(FontFamilyNameArgumentDescription);
+ WINGET_DEFINE_RESOURCE_STRINGID(FontFilePaths);
+ WINGET_DEFINE_RESOURCE_STRINGID(FontListCommandLongDescription);
+ WINGET_DEFINE_RESOURCE_STRINGID(FontListCommandShortDescription);
+ WINGET_DEFINE_RESOURCE_STRINGID(FontVersion);
WINGET_DEFINE_RESOURCE_STRINGID(ForceArgumentDescription);
WINGET_DEFINE_RESOURCE_STRINGID(GatedVersionArgumentDescription);
WINGET_DEFINE_RESOURCE_STRINGID(GetManifestResultVersionNotFound);
@@ -403,6 +413,7 @@ namespace AppInstaller::CLI::Resource
WINGET_DEFINE_RESOURCE_STRINGID(NoAdminRepairForUserScopePackage);
WINGET_DEFINE_RESOURCE_STRINGID(NoApplicableInstallers);
WINGET_DEFINE_RESOURCE_STRINGID(NoExperimentalFeaturesMessage);
+ WINGET_DEFINE_RESOURCE_STRINGID(NoInstalledFontFound);
WINGET_DEFINE_RESOURCE_STRINGID(NoInstalledPackageFound);
WINGET_DEFINE_RESOURCE_STRINGID(NoPackageFound);
WINGET_DEFINE_RESOURCE_STRINGID(NoPackageSelectionArgumentProvided);
diff --git a/src/AppInstallerCLICore/Workflows/FontFlow.cpp b/src/AppInstallerCLICore/Workflows/FontFlow.cpp
new file mode 100644
index 0000000000..ce9bb7a233
--- /dev/null
+++ b/src/AppInstallerCLICore/Workflows/FontFlow.cpp
@@ -0,0 +1,124 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+#include "pch.h"
+#include "FontFlow.h"
+#include "TableOutput.h"
+#include
+#include
+
+namespace AppInstaller::CLI::Workflow
+{
+ using namespace AppInstaller::CLI::Execution;
+
+ namespace
+ {
+ struct InstalledFontFamiliesTableLine
+ {
+ InstalledFontFamiliesTableLine(Utility::LocIndString familyName, int faceCount)
+ : FamilyName(familyName), FaceCount(faceCount) {}
+
+ Utility::LocIndString FamilyName;
+ int FaceCount;
+ };
+
+ struct InstalledFontFacesTableLine
+ {
+ InstalledFontFacesTableLine(Utility::LocIndString familyName, Utility::LocIndString faceName, Utility::LocIndString faceVersion, std::filesystem::path filePath)
+ : FamilyName(familyName), FaceName(faceName), FaceVersion(faceVersion), FilePath(filePath) {}
+
+ Utility::LocIndString FamilyName;
+ Utility::LocIndString FaceName;
+ Utility::LocIndString FaceVersion;
+ std::filesystem::path FilePath;
+ };
+
+ void OutputInstalledFontFamiliesTable(Execution::Context& context, const std::vector& lines)
+ {
+ Execution::TableOutput<2> table(context.Reporter, { Resource::String::FontFamily, Resource::String::FontFaces });
+
+ for (auto line : lines)
+ {
+ table.OutputLine({ line.FamilyName, std::to_string(line.FaceCount) });
+ }
+
+ table.Complete();
+ }
+
+ void OutputInstalledFontFacesTable(Execution::Context& context, const std::vector& lines)
+ {
+ Execution::TableOutput<4> table(context.Reporter, { Resource::String::FontFamily, Resource::String::FontFace, Resource::String::FontVersion, Resource::String::FontFilePaths });
+
+ bool anonymizePath = Settings::User().Get();
+
+ for (auto line : lines)
+ {
+ if (anonymizePath)
+ {
+ AppInstaller::Runtime::ReplaceProfilePathsWithEnvironmentVariable(line.FilePath);
+ }
+
+ table.OutputLine({ line.FamilyName, line.FaceName, line.FaceVersion, line.FilePath.u8string() });
+ }
+
+ table.Complete();
+ }
+ }
+
+ void ReportInstalledFonts(Execution::Context& context)
+ {
+ Fonts::FontCatalog fontCatalog;
+
+ if (context.Args.Contains(Args::Type::Family))
+ {
+ // TODO: Create custom source and search mechanism for fonts.
+ const auto& familyNameArg = AppInstaller::Utility::ConvertToUTF16(context.Args.GetArg(Args::Type::Family));
+ const auto& fontFamilies = fontCatalog.GetInstalledFontFamilies(familyNameArg);
+
+ if (fontFamilies.empty())
+ {
+ context.Reporter.Info() << Resource::String::NoInstalledFontFound << std::endl;
+ return;
+ }
+
+ std::vector lines;
+
+ for (const auto& fontFamily : fontFamilies)
+ {
+ const auto& familyName = Utility::LocIndString(Utility::ConvertToUTF8(fontFamily.Name));
+
+ for (const auto& fontFace : fontFamily.Faces)
+ {
+ for (const auto& filePath : fontFace.FilePaths)
+ {
+ InstalledFontFacesTableLine line(
+ familyName,
+ Utility::LocIndString(Utility::ToLower(Utility::ConvertToUTF8(fontFace.Name))),
+ Utility::LocIndString(fontFace.Version.ToString()),
+ filePath.u8string());
+
+ lines.push_back(std::move(line));
+ }
+ }
+ }
+
+ OutputInstalledFontFacesTable(context, lines);
+ }
+ else
+ {
+ const auto& fontFamilies = fontCatalog.GetInstalledFontFamilies();
+ std::vector lines;
+
+ for (const auto& fontFamily : fontFamilies)
+ {
+ InstalledFontFamiliesTableLine line(
+ Utility::LocIndString(Utility::ConvertToUTF8(fontFamily.Name)),
+ static_cast(fontFamily.Faces.size())
+ );
+
+ lines.push_back(std::move(line));
+ }
+
+ OutputInstalledFontFamiliesTable(context, lines);
+ }
+ }
+}
diff --git a/src/AppInstallerCLICore/Workflows/FontFlow.h b/src/AppInstallerCLICore/Workflows/FontFlow.h
new file mode 100644
index 0000000000..c3346e1a60
--- /dev/null
+++ b/src/AppInstallerCLICore/Workflows/FontFlow.h
@@ -0,0 +1,13 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+#pragma once
+#include "ExecutionContext.h"
+
+namespace AppInstaller::CLI::Workflow
+{
+ // Reports the installed fonts as a table.
+ // Required Args: None
+ // Inputs: None
+ // Outputs: None
+ void ReportInstalledFonts(Execution::Context& context);
+}
diff --git a/src/AppInstallerCLIE2ETests/BaseCommand.cs b/src/AppInstallerCLIE2ETests/BaseCommand.cs
index 1b28227fd5..faa07c2dd0 100644
--- a/src/AppInstallerCLIE2ETests/BaseCommand.cs
+++ b/src/AppInstallerCLIE2ETests/BaseCommand.cs
@@ -1,4 +1,4 @@
-// -----------------------------------------------------------------------------
+// -----------------------------------------------------------------------------
//
// Copyright (c) Microsoft Corporation. Licensed under the MIT License.
//
@@ -6,10 +6,7 @@
namespace AppInstallerCLIE2ETests
{
- using System;
- using System.IO;
using AppInstallerCLIE2ETests.Helpers;
- using Newtonsoft.Json.Linq;
using NUnit.Framework;
///
diff --git a/src/AppInstallerCLIPackage/Shared/Strings/en-us/winget.resw b/src/AppInstallerCLIPackage/Shared/Strings/en-us/winget.resw
index 9f0f88b8df..46d093b80c 100644
--- a/src/AppInstallerCLIPackage/Shared/Strings/en-us/winget.resw
+++ b/src/AppInstallerCLIPackage/Shared/Strings/en-us/winget.resw
@@ -3138,5 +3138,40 @@ Please specify one of them using the --source option to proceed.
Downloaded zero byte installer; ensure that your network connection is working properly.
+
+
+ Manage fonts
+
+
+ Manage fonts with sub-commands. Fonts can be installed, upgraded, or uninstalled similar to packages.
+
+
+ Family
+
+
+ Faces
+ "Faces" represents the typeface of the font family such as 'Bold' or 'Italic'
+
+
+ Filter results by family name
+
+
+ Face
+ "Face" represents the typeface of a font family such as 'Bold' or 'Italic'
+
+
+ Paths
+
+
+ No installed font found matching input criteria.
+
+
+ List installed fonts
+
+
+ List all installed fonts, or full details of a specific font.
+
+
+ Version
\ No newline at end of file
diff --git a/src/AppInstallerCLITests/AppInstallerCLITests.vcxproj b/src/AppInstallerCLITests/AppInstallerCLITests.vcxproj
index 043eab3f5a..7500b2c1bb 100644
--- a/src/AppInstallerCLITests/AppInstallerCLITests.vcxproj
+++ b/src/AppInstallerCLITests/AppInstallerCLITests.vcxproj
@@ -284,6 +284,7 @@
+
diff --git a/src/AppInstallerCLITests/AppInstallerCLITests.vcxproj.filters b/src/AppInstallerCLITests/AppInstallerCLITests.vcxproj.filters
index a24784a802..5d28140d17 100644
--- a/src/AppInstallerCLITests/AppInstallerCLITests.vcxproj.filters
+++ b/src/AppInstallerCLITests/AppInstallerCLITests.vcxproj.filters
@@ -368,6 +368,9 @@
Source Files\Repository
+
+ Source Files\Common
+
diff --git a/src/AppInstallerCLITests/Fonts.cpp b/src/AppInstallerCLITests/Fonts.cpp
new file mode 100644
index 0000000000..81a50791e8
--- /dev/null
+++ b/src/AppInstallerCLITests/Fonts.cpp
@@ -0,0 +1,37 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+#include "pch.h"
+#include "TestCommon.h"
+#include
+
+using namespace AppInstaller::Fonts;
+using namespace TestCommon;
+
+constexpr std::wstring_view s_testFontName = L"Times New Roman";
+
+TEST_CASE("GetInstalledFonts", "[fonts]")
+{
+ FontCatalog fontCatalog;
+ std::vector installedFontFamilies;
+ REQUIRE_NOTHROW(installedFontFamilies = fontCatalog.GetInstalledFontFamilies());
+ REQUIRE(installedFontFamilies.size() > 0);
+}
+
+TEST_CASE("GetSingleFontFamily", "[fonts]")
+{
+ FontCatalog fontCatalog;
+ std::vector fontFamily;
+ REQUIRE_NOTHROW(fontFamily = fontCatalog.GetInstalledFontFamilies(std::wstring(s_testFontName)));
+ REQUIRE_FALSE(fontFamily.empty());
+ FontFamily singleFontFamily = fontFamily[0];
+ REQUIRE(AppInstaller::Utility::CaseInsensitiveEquals(singleFontFamily.Name, s_testFontName));
+ REQUIRE(singleFontFamily.Faces.size() > 0);
+}
+
+TEST_CASE("GetInvalidFontFamily", "[fonts]")
+{
+ FontCatalog fontCatalog;
+ std::vector fontFamily;
+ REQUIRE_NOTHROW(fontFamily = fontCatalog.GetInstalledFontFamilies(L"Invalid Font"));
+ REQUIRE(fontFamily.empty());
+}
diff --git a/src/AppInstallerCLITests/Versions.cpp b/src/AppInstallerCLITests/Versions.cpp
index 0988d58074..54975eda49 100644
--- a/src/AppInstallerCLITests/Versions.cpp
+++ b/src/AppInstallerCLITests/Versions.cpp
@@ -421,3 +421,30 @@ TEST_CASE("SemanticVersion", "[versions]")
REQUIRE(version.BuildMetadata() == Version("4.5.6"));
REQUIRE(version.PartAt(2).Other == "-beta+4.5.6");
}
+
+TEST_CASE("OpenTypeFontVersion", "[versions]")
+{
+ // Valid font version.
+ OpenTypeFontVersion version = OpenTypeFontVersion("Version 1.234");
+ REQUIRE(version.ToString() == "1.234");
+ REQUIRE(version.GetParts().size() == 2);
+ REQUIRE(version.PartAt(0).Integer == 1);
+ REQUIRE(version.PartAt(1).Integer == 234);
+
+ // Font version with additional metadata.
+ version = OpenTypeFontVersion("Version 9.876.54 ;2024");
+ REQUIRE(version.ToString() == "9.876");
+ REQUIRE(version.GetParts().size() == 2);
+ REQUIRE(version.PartAt(0).Integer == 9);
+ REQUIRE(version.PartAt(1).Integer == 876);
+
+ // Invalid version. Font version must have at least 2 parts.
+ REQUIRE_NOTHROW(version = OpenTypeFontVersion("1234567"));
+ REQUIRE(version.IsUnknown());
+ REQUIRE(version.ToString() == "Unknown");
+
+ // Major and minor parts must have digits.
+ REQUIRE_NOTHROW(version = OpenTypeFontVersion(" abc.def "));
+ REQUIRE(version.IsUnknown());
+ REQUIRE(version.ToString() == "Unknown");
+}
diff --git a/src/AppInstallerCommonCore/AppInstallerCommonCore.vcxproj b/src/AppInstallerCommonCore/AppInstallerCommonCore.vcxproj
index d57fcfc5e6..927547fc8f 100644
--- a/src/AppInstallerCommonCore/AppInstallerCommonCore.vcxproj
+++ b/src/AppInstallerCommonCore/AppInstallerCommonCore.vcxproj
@@ -427,6 +427,7 @@
+
true
@@ -478,6 +479,7 @@
+
true
diff --git a/src/AppInstallerCommonCore/AppInstallerCommonCore.vcxproj.filters b/src/AppInstallerCommonCore/AppInstallerCommonCore.vcxproj.filters
index 3498d88dfb..1cb64a4d0e 100644
--- a/src/AppInstallerCommonCore/AppInstallerCommonCore.vcxproj.filters
+++ b/src/AppInstallerCommonCore/AppInstallerCommonCore.vcxproj.filters
@@ -204,6 +204,9 @@
Public\winget
+
+ Public\winget
+
@@ -368,6 +371,9 @@
Source Files
+
+ Source Files
+
diff --git a/src/AppInstallerCommonCore/ExperimentalFeature.cpp b/src/AppInstallerCommonCore/ExperimentalFeature.cpp
index 733ff9ccd5..c3709c6802 100644
--- a/src/AppInstallerCommonCore/ExperimentalFeature.cpp
+++ b/src/AppInstallerCommonCore/ExperimentalFeature.cpp
@@ -48,6 +48,8 @@ namespace AppInstaller::Settings
return userSettings.Get();
case ExperimentalFeature::Feature::ConfigureExport:
return userSettings.Get();
+ case ExperimentalFeature::Feature::Font:
+ return userSettings.Get();
default:
THROW_HR(E_UNEXPECTED);
}
@@ -85,6 +87,9 @@ namespace AppInstaller::Settings
return ExperimentalFeature{ "Configure Self Elevation", "configureSelfElevate", "https://aka.ms/winget-settings", Feature::ConfigureSelfElevation };
case Feature::ConfigureExport:
return ExperimentalFeature{ "Configure Export", "configureExport", "https://aka.ms/winget-settings", Feature::ConfigureExport };
+ case Feature::Font:
+ return ExperimentalFeature{ "Font", "Font", "https://aka.ms/winget-settings", Feature::Font };
+
default:
THROW_HR(E_UNEXPECTED);
}
diff --git a/src/AppInstallerCommonCore/Fonts.cpp b/src/AppInstallerCommonCore/Fonts.cpp
new file mode 100644
index 0000000000..6f5f328f93
--- /dev/null
+++ b/src/AppInstallerCommonCore/Fonts.cpp
@@ -0,0 +1,175 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+#include "pch.h"
+#include
+#include
+#include
+
+namespace AppInstaller::Fonts
+{
+ namespace
+ {
+ std::vector GetFontFilePaths(const wil::com_ptr& fontFace)
+ {
+ UINT32 fileCount = 0;
+ THROW_IF_FAILED(fontFace->GetFiles(&fileCount, nullptr));
+
+ static_assert(sizeof(wil::com_ptr) == sizeof(IDWriteFontFile*));
+ std::vector> fontFiles;
+ fontFiles.resize(fileCount);
+
+ THROW_IF_FAILED(fontFace->GetFiles(&fileCount, fontFiles[0].addressof()));
+
+ std::vector filePaths;
+ for (UINT32 i = 0; i < fileCount; ++i) {
+ wil::com_ptr loader;
+ THROW_IF_FAILED(fontFiles[i]->GetLoader(loader.addressof()));
+
+ const void* fontFileReferenceKey;
+ UINT32 fontFileReferenceKeySize;
+ THROW_IF_FAILED(fontFiles[i]->GetReferenceKey(&fontFileReferenceKey, &fontFileReferenceKeySize));
+
+ if (const auto localLoader = loader.try_query()) {
+ UINT32 pathLength;
+ THROW_IF_FAILED(localLoader->GetFilePathLengthFromKey(fontFileReferenceKey, fontFileReferenceKeySize, &pathLength));
+ pathLength += 1; // Account for the trailing null terminator during allocation.
+
+ std::wstring path;
+ path.resize(pathLength);
+ THROW_IF_FAILED(localLoader->GetFilePathFromKey(fontFileReferenceKey, fontFileReferenceKeySize, &path[0], pathLength));
+ path.resize(pathLength - 1); // Remove the null char.
+ filePaths.emplace_back(std::move(path));
+ }
+ }
+
+ return filePaths;
+ }
+ }
+
+ FontCatalog::FontCatalog()
+ {
+ m_preferredLocales = AppInstaller::Locale::GetUserPreferredLanguagesUTF16();
+ }
+
+ std::vector FontCatalog::GetInstalledFontFamilies(std::optional familyName)
+ {
+ wil::com_ptr factory;
+ THROW_IF_FAILED(DWriteCreateFactory(DWRITE_FACTORY_TYPE_SHARED, __uuidof(factory), factory.put_unknown()));
+
+ wil::com_ptr collection;
+ THROW_IF_FAILED(factory->GetSystemFontCollection(collection.addressof(), FALSE));
+
+ std::vector installedFontFamilies;
+
+ if (familyName.has_value())
+ {
+ UINT32 index;
+ BOOL exists;
+ THROW_IF_FAILED(collection->FindFamilyName(familyName.value().c_str(), &index, &exists));
+
+ if (exists)
+ {
+ installedFontFamilies.emplace_back(GetFontFamilyByIndex(collection, index));
+ }
+ }
+ else
+ {
+ UINT32 familyCount = collection->GetFontFamilyCount();
+
+ for (UINT32 index = 0; index < familyCount; index++)
+ {
+ installedFontFamilies.emplace_back(GetFontFamilyByIndex(collection, index));
+ }
+ }
+
+ return installedFontFamilies;
+ }
+
+ std::wstring FontCatalog::GetLocalizedStringFromFont(const wil::com_ptr& localizedStringCollection)
+ {
+ UINT32 index = 0;
+ BOOL exists = false;
+
+ for (const auto& locale : m_preferredLocales)
+ {
+ if (SUCCEEDED_LOG(localizedStringCollection->FindLocaleName(locale.c_str(), &index, &exists)) && exists)
+ {
+ break;
+ }
+ }
+
+ // If the locale does not exist, resort to the default value at the 0 index.
+ if (!exists)
+ {
+ index = 0;
+ }
+
+ UINT32 length = 0;
+ THROW_IF_FAILED(localizedStringCollection->GetStringLength(index, &length));
+ length += 1; // Account for the trailing null terminator during allocation.
+
+ std::wstring localizedString;
+ localizedString.resize(length);
+ THROW_IF_FAILED(localizedStringCollection->GetString(index, &localizedString[0], length));
+ localizedString.resize(length - 1); // Remove the null char.
+ return localizedString;
+ }
+
+ std::wstring FontCatalog::GetFontFaceName(const wil::com_ptr& font)
+ {
+ wil::com_ptr faceNames;
+ THROW_IF_FAILED(font->GetFaceNames(faceNames.addressof()));
+ return GetLocalizedStringFromFont(faceNames);
+ }
+
+ std::wstring FontCatalog::GetFontFamilyName(const wil::com_ptr& fontFamily)
+ {
+ wil::com_ptr familyNames;
+ THROW_IF_FAILED(fontFamily->GetFamilyNames(familyNames.addressof()));
+ return GetLocalizedStringFromFont(familyNames);
+ }
+
+ Utility::OpenTypeFontVersion FontCatalog::GetFontFaceVersion(const wil::com_ptr& font)
+ {
+ wil::com_ptr fontVersion;
+ BOOL exists;
+ THROW_IF_FAILED(font->GetInformationalStrings(DWRITE_INFORMATIONAL_STRING_VERSION_STRINGS, fontVersion.addressof(), &exists));
+ if (!exists)
+ {
+ return {};
+ }
+
+ std::string value = AppInstaller::Utility::ConvertToUTF8(GetLocalizedStringFromFont(fontVersion));
+ Utility::OpenTypeFontVersion openTypeFontVersion{ value };
+ return openTypeFontVersion;
+ }
+
+ FontFamily FontCatalog::GetFontFamilyByIndex(const wil::com_ptr& collection, UINT32 index)
+ {
+ wil::com_ptr family;
+ THROW_IF_FAILED(collection->GetFontFamily(index, family.addressof()));
+ std::wstring familyName = GetFontFamilyName(family);
+
+ std::vector fontFaces;
+ UINT32 fontCount = family->GetFontCount();
+ for (UINT32 fontIndex = 0; fontIndex < fontCount; fontIndex++)
+ {
+ wil::com_ptr font;
+ THROW_IF_FAILED(family->GetFont(fontIndex, font.addressof()));
+
+ wil::com_ptr fontFace;
+ THROW_IF_FAILED(font->CreateFontFace(fontFace.addressof()));
+
+ FontFace fontFaceEntry;
+ fontFaceEntry.Name = GetFontFaceName(font);
+ fontFaceEntry.Version = GetFontFaceVersion(font);
+ fontFaceEntry.FilePaths = GetFontFilePaths(fontFace);
+ fontFaces.emplace_back(std::move(fontFaceEntry));
+ }
+
+ FontFamily fontFamily;
+ fontFamily.Name = std::move(familyName);
+ fontFamily.Faces = std::move(fontFaces);
+ return fontFamily;
+ }
+}
diff --git a/src/AppInstallerCommonCore/Locale.cpp b/src/AppInstallerCommonCore/Locale.cpp
index c308584226..ae8985910c 100644
--- a/src/AppInstallerCommonCore/Locale.cpp
+++ b/src/AppInstallerCommonCore/Locale.cpp
@@ -120,6 +120,18 @@ namespace AppInstaller::Locale
return result;
}
+ std::vector GetUserPreferredLanguagesUTF16()
+ {
+ std::vector result;
+
+ for (const auto& lang : winrt::Windows::System::UserProfile::GlobalizationPreferences::Languages())
+ {
+ result.emplace_back(std::wstring(lang));
+ }
+
+ return result;
+ }
+
std::string LocaleIdToBcp47Tag(LCID localeId)
{
WCHAR localeName[MAX_LOCALE_SNAME_LEN] = {0};
diff --git a/src/AppInstallerCommonCore/Public/AppInstallerRuntime.h b/src/AppInstallerCommonCore/Public/AppInstallerRuntime.h
index 2b8a729a2d..cecadb24d8 100644
--- a/src/AppInstallerCommonCore/Public/AppInstallerRuntime.h
+++ b/src/AppInstallerCommonCore/Public/AppInstallerRuntime.h
@@ -56,6 +56,10 @@ namespace AppInstaller::Runtime
CLIExecutable,
// The location of the image assets, if it exists.
ImageAssets,
+ // The location where fonts are installed with user scope.
+ FontsUserInstallLocation,
+ // The location where fonts are installed with machine scope.
+ FontsMachineInstallLocation,
// Always one more than the last path; for being able to iterate paths in tests.
Max
};
@@ -70,6 +74,9 @@ namespace AppInstaller::Runtime
return Filesystem::GetPathTo(path, forDisplay);
}
+ // Replaces the substring in the path with the user profile environment variable.
+ void ReplaceProfilePathsWithEnvironmentVariable(std::filesystem::path& path);
+
// Gets a new temp file path.
std::filesystem::path GetNewTempFilePath();
diff --git a/src/AppInstallerCommonCore/Public/winget/ExperimentalFeature.h b/src/AppInstallerCommonCore/Public/winget/ExperimentalFeature.h
index 733f1aa453..80469aa24c 100644
--- a/src/AppInstallerCommonCore/Public/winget/ExperimentalFeature.h
+++ b/src/AppInstallerCommonCore/Public/winget/ExperimentalFeature.h
@@ -27,6 +27,7 @@ namespace AppInstaller::Settings
Configuration03 = 0x4,
ConfigureSelfElevation = 0x8,
ConfigureExport = 0x10,
+ Font = 0x20,
Max, // This MUST always be after all experimental features
// Features listed after Max will not be shown with the features command
diff --git a/src/AppInstallerCommonCore/Public/winget/Fonts.h b/src/AppInstallerCommonCore/Public/winget/Fonts.h
new file mode 100644
index 0000000000..443dea5f7a
--- /dev/null
+++ b/src/AppInstallerCommonCore/Public/winget/Fonts.h
@@ -0,0 +1,39 @@
+// Copyright (c) Microsoft Corporation.
+// Licensed under the MIT License.
+#pragma once
+#include
+#include
+#include
+
+namespace AppInstaller::Fonts
+{
+ struct FontFace
+ {
+ std::wstring Name;
+ std::vector FilePaths;
+ Utility::OpenTypeFontVersion Version;
+ };
+
+ struct FontFamily
+ {
+ std::wstring Name;
+ std::vector Faces;
+ };
+
+ struct FontCatalog
+ {
+ FontCatalog();
+
+ // Gets all installed font families on the system. If an exact family name is provided and found, returns the installed font family.
+ std::vector GetInstalledFontFamilies(std::optional familyName = {});
+
+ private:
+ FontFamily GetFontFamilyByIndex(const wil::com_ptr& collection, UINT32 index);
+ std::wstring GetLocalizedStringFromFont(const wil::com_ptr& localizedStringCollection);
+ std::wstring GetFontFamilyName(const wil::com_ptr& fontFamily);
+ std::wstring GetFontFaceName(const wil::com_ptr& font);
+ Utility::OpenTypeFontVersion GetFontFaceVersion(const wil::com_ptr& font);
+
+ std::vector m_preferredLocales;
+ };
+}
diff --git a/src/AppInstallerCommonCore/Public/winget/Locale.h b/src/AppInstallerCommonCore/Public/winget/Locale.h
index f4cd7a2d54..5ae86a8de4 100644
--- a/src/AppInstallerCommonCore/Public/winget/Locale.h
+++ b/src/AppInstallerCommonCore/Public/winget/Locale.h
@@ -20,6 +20,9 @@ namespace AppInstaller::Locale
// Get the list of user Preferred Languages from settings. Returns an empty vector in rare cases of failure.
std::vector GetUserPreferredLanguages();
+ // Get the list of user Preferred Languages from settings. Returns an empty vector in rare cases of failure.
+ std::vector GetUserPreferredLanguagesUTF16();
+
// Get the bcp47 tag from a locale id. Returns empty string if conversion cannot be performed.
std::string LocaleIdToBcp47Tag(LCID localeId);
-}
\ No newline at end of file
+}
diff --git a/src/AppInstallerCommonCore/Public/winget/UserSettings.h b/src/AppInstallerCommonCore/Public/winget/UserSettings.h
index 5d8615485a..b580775c28 100644
--- a/src/AppInstallerCommonCore/Public/winget/UserSettings.h
+++ b/src/AppInstallerCommonCore/Public/winget/UserSettings.h
@@ -78,6 +78,7 @@ namespace AppInstaller::Settings
EFConfiguration03,
EFConfigureSelfElevation,
EFConfigureExport,
+ EFFonts,
// Telemetry
TelemetryDisable,
// Install behavior
@@ -161,6 +162,7 @@ namespace AppInstaller::Settings
SETTINGMAPPING_SPECIALIZATION(Setting::EFConfiguration03, bool, bool, false, ".experimentalFeatures.configuration03"sv);
SETTINGMAPPING_SPECIALIZATION(Setting::EFConfigureSelfElevation, bool, bool, false, ".experimentalFeatures.configureSelfElevate"sv);
SETTINGMAPPING_SPECIALIZATION(Setting::EFConfigureExport, bool, bool, false, ".experimentalFeatures.configureExport"sv);
+ SETTINGMAPPING_SPECIALIZATION(Setting::EFFonts, bool, bool, false, ".experimentalFeatures.fonts"sv);
// Telemetry
SETTINGMAPPING_SPECIALIZATION(Setting::TelemetryDisable, bool, bool, false, ".telemetry.disable"sv);
// Install behavior
diff --git a/src/AppInstallerCommonCore/Runtime.cpp b/src/AppInstallerCommonCore/Runtime.cpp
index 92bb493a83..3a8973d8e9 100644
--- a/src/AppInstallerCommonCore/Runtime.cpp
+++ b/src/AppInstallerCommonCore/Runtime.cpp
@@ -29,6 +29,7 @@ namespace AppInstaller::Runtime
constexpr std::string_view s_PortablePackageRoot = "WinGet"sv;
constexpr std::string_view s_PortablePackagesDirectory = "Packages"sv;
constexpr std::string_view s_LinksDirectory = "Links"sv;
+ constexpr std::string_view s_FontsInstallDirectory = "Microsoft\\Windows\\Fonts"sv;
// Use production CLSIDs as a surrogate for repository location.
#if USE_PROD_CLSIDS
constexpr std::string_view s_ImageAssetsDirectoryRelative = "Assets\\WinGet"sv;
@@ -162,15 +163,6 @@ namespace AppInstaller::Runtime
return result;
}
-
- // Try to replace LOCALAPPDATA first as it is the likely location, fall back to trying USERPROFILE.
- void ReplaceProfilePathsWithEnvironmentVariable(std::filesystem::path& path)
- {
- if (!ReplaceCommonPathPrefix(path, GetKnownFolderPath(FOLDERID_LocalAppData), s_LocalAppDataEnvironmentVariable))
- {
- ReplaceCommonPathPrefix(path, GetKnownFolderPath(FOLDERID_Profile), s_UserProfileEnvironmentVariable);
- }
- }
}
void SetRuntimePathStateName(std::string name)
@@ -240,6 +232,14 @@ namespace AppInstaller::Runtime
result.Path = GetKnownFolderPath(FOLDERID_Downloads);
mayBeInProfilePath = true;
break;
+ case PathName::FontsUserInstallLocation:
+ result.Path = GetKnownFolderPath(FOLDERID_LocalAppData);
+ result.Path /= s_FontsInstallDirectory;
+ mayBeInProfilePath = true;
+ break;
+ case PathName::FontsMachineInstallLocation:
+ result.Path = GetKnownFolderPath(FOLDERID_Fonts);
+ break;
default:
THROW_HR(E_UNEXPECTED);
}
@@ -314,6 +314,8 @@ namespace AppInstaller::Runtime
case PathName::PortableLinksUserLocation:
case PathName::PortablePackageUserRoot:
case PathName::UserProfileDownloads:
+ case PathName::FontsUserInstallLocation:
+ case PathName::FontsMachineInstallLocation:
result = GetPathDetailsCommon(path, forDisplay);
break;
case PathName::SelfPackageRoot:
@@ -418,6 +420,8 @@ namespace AppInstaller::Runtime
case PathName::PortableLinksUserLocation:
case PathName::PortablePackageUserRoot:
case PathName::UserProfileDownloads:
+ case PathName::FontsUserInstallLocation:
+ case PathName::FontsMachineInstallLocation:
result = GetPathDetailsCommon(path, forDisplay);
break;
case PathName::SelfPackageRoot:
@@ -476,6 +480,15 @@ namespace AppInstaller::Runtime
return result;
}
+ // Try to replace LOCALAPPDATA first as it is the likely location, fall back to trying USERPROFILE.
+ void ReplaceProfilePathsWithEnvironmentVariable(std::filesystem::path& path)
+ {
+ if (!ReplaceCommonPathPrefix(path, GetKnownFolderPath(FOLDERID_LocalAppData), s_LocalAppDataEnvironmentVariable))
+ {
+ ReplaceCommonPathPrefix(path, GetKnownFolderPath(FOLDERID_Profile), s_UserProfileEnvironmentVariable);
+ }
+ }
+
std::filesystem::path GetNewTempFilePath()
{
GUID guid;
diff --git a/src/AppInstallerCommonCore/UserSettings.cpp b/src/AppInstallerCommonCore/UserSettings.cpp
index 476971d2f0..9e13610537 100644
--- a/src/AppInstallerCommonCore/UserSettings.cpp
+++ b/src/AppInstallerCommonCore/UserSettings.cpp
@@ -269,6 +269,7 @@ namespace AppInstaller::Settings
WINGET_VALIDATE_PASS_THROUGH(EFConfiguration03)
WINGET_VALIDATE_PASS_THROUGH(EFConfigureSelfElevation)
WINGET_VALIDATE_PASS_THROUGH(EFConfigureExport)
+ WINGET_VALIDATE_PASS_THROUGH(EFFonts)
WINGET_VALIDATE_PASS_THROUGH(AnonymizePathForDisplay)
WINGET_VALIDATE_PASS_THROUGH(TelemetryDisable)
WINGET_VALIDATE_PASS_THROUGH(InteractivityDisable)
diff --git a/src/AppInstallerSharedLib/Public/AppInstallerVersions.h b/src/AppInstallerSharedLib/Public/AppInstallerVersions.h
index 569a33aa32..9df7cab413 100644
--- a/src/AppInstallerSharedLib/Public/AppInstallerVersions.h
+++ b/src/AppInstallerSharedLib/Public/AppInstallerVersions.h
@@ -281,4 +281,20 @@ namespace AppInstaller::Utility
// Checks if there are overlaps within the list of version ranges
bool HasOverlapInVersionRanges(const std::vector& ranges);
+
+ // The OpenType font version.
+ // The format of this version type is 'Version 1.234 ;567'
+ // The only part that is of importance is the 'Major.Minor' parts.
+ // The 'Version' string is typically found at the beginning of the version string.
+ // Any value after a digit that is not a '.' represents some other meaning.
+ struct OpenTypeFontVersion : Version
+ {
+ OpenTypeFontVersion() = default;
+
+ OpenTypeFontVersion(std::string&& version);
+ OpenTypeFontVersion(const std::string& version) :
+ OpenTypeFontVersion(std::string(version)) {}
+
+ void Assign(std::string version, std::string_view splitChars = DefaultSplitChars) override;
+ };
}
diff --git a/src/AppInstallerSharedLib/Versions.cpp b/src/AppInstallerSharedLib/Versions.cpp
index 6b02675d1d..2c9972c561 100644
--- a/src/AppInstallerSharedLib/Versions.cpp
+++ b/src/AppInstallerSharedLib/Versions.cpp
@@ -661,4 +661,64 @@ namespace AppInstaller::Utility
return false;
}
+
+ OpenTypeFontVersion::OpenTypeFontVersion(std::string&& version)
+ {
+ Assign(std::move(version), DefaultSplitChars);
+ }
+
+ void OpenTypeFontVersion::Assign(std::string version, std::string_view splitChars)
+ {
+ // Open type version requires using the default split character
+ THROW_HR_IF(E_INVALIDARG, splitChars != DefaultSplitChars);
+
+ // Split on default split character.
+ std::vector parts = Split(version, '.', true);
+
+ std::string majorString;
+ std::string minorString;
+
+ // Font version must have a "major.minor" part.
+ if (parts.size() >= 2)
+ {
+ // Find first digit and trim all preceding characters.
+ std::string firstPart = parts[0];
+ size_t majorStartIndex = firstPart.find_first_of(s_Digit_Characters);
+
+ if (majorStartIndex != std::string::npos)
+ {
+ firstPart.erase(0, majorStartIndex);
+ }
+
+ size_t majorEndIndex = firstPart.find_last_of(s_Digit_Characters);
+ majorString = firstPart.substr(0, majorEndIndex + 1);
+
+ // Parse and verify minor part.
+ std::string secondPart = parts[1];
+ size_t endPos = secondPart.find_first_not_of(s_Digit_Characters);
+
+ // If a non-digit character exists, trim off the remainder.
+ if (endPos != std::string::npos)
+ {
+ secondPart.erase(endPos, secondPart.length());
+ }
+
+ minorString = secondPart;
+ }
+
+ // Verify results.
+ if (!majorString.empty() && !minorString.empty())
+ {
+ m_parts.emplace_back(majorString);
+ m_parts.emplace_back(minorString);
+ m_version = Utility::Join(DefaultSplitChars, { majorString, minorString });
+
+ Trim();
+ }
+ else
+ {
+ m_version = s_Version_Part_Unknown;
+ m_parts.emplace_back(0, std::string{ s_Version_Part_Unknown });
+ }
+ }
}