mirror of
https://github.com/microsoft/terminal.git
synced 2026-04-03 17:00:23 -04:00
Use GetAttribute('x:Name') instead of .Name in GenerateSettingsIndex.ps1
to avoid PowerShell's XML integration returning the element tag name
(e.g. 'local:SettingContainer') when x:Name is absent.
Also add missing x:Name attributes to:
- Compatibility.xaml: AmbiguousWidth SettingContainer
- NewTabMenu.xaml: AddRemainingProfiles and CurrentFolderIcon containers
## Summary of the Pull Request
Fix GenerateSettingsIndex.ps1 emitting "local:SettingContainer" as the
element name in the generated index when a SettingContainer has no
x:Name attribute.
## References and Relevant Issues
Per DHowett's comment - the root cause is PowerShell's XML integration:
$element.Name returns the XML element tag name (e.g.
local:SettingContainer) when no x:Name attribute exists.
## Detailed Description of the Pull Request / Additional comments
Two changes:
- GenerateSettingsIndex.ps1: Replace $settingContainer.Name with
$settingContainer.GetAttribute("x:Name"), which correctly returns an
empty string when the attribute is absent instead of the element tag
name.
- Add missing x:Name attributes to three SettingContainer elements:
- Compatibility.xaml: AmbiguousWidth (Globals_AmbiguousWidth)
- NewTabMenu.xaml: AddRemainingProfiles
(NewTabMenu_AddRemainingProfiles)
- NewTabMenu.xaml: CurrentFolderIcon (NewTabMenu_CurrentFolderIcon)
- This fixes four incorrect IndexEntry lines in the generated output
that previously contained L"local:SettingContainer" as the element name.
## Validation Steps Performed
Ran GenerateSettingsIndex.ps1 before and after - confirmed the four
incorrect entries with L"local:SettingContainer" are now generated with
the correct x:Name values (or empty string where appropriate).
## PR Checklist
Closes #19929
Co-authored-by: Sagar Bhure <sagarbhure@microsoft.com>
461 lines
17 KiB
PowerShell
461 lines
17 KiB
PowerShell
<#
|
|
Copyright (c) Microsoft Corporation.
|
|
Licensed under the MIT license.
|
|
.SYNOPSIS
|
|
Scans XAML files for local:SettingContainer entries and generates GeneratedSettingsIndex.g.h / .g.cpp.
|
|
|
|
.PARAMETER SourceDir
|
|
Directory to scan recursively for .xaml files.
|
|
|
|
.PARAMETER OutputDir
|
|
Directory to place generated C++ files.
|
|
#>
|
|
[CmdletBinding()]
|
|
param(
|
|
[Parameter(Mandatory=$false)][string]$SourceDir = "$PSScriptRoot\..\src\cascadia\TerminalSettingsEditor\",
|
|
[Parameter(Mandatory=$false)][string]$OutputDir = "$PSScriptRoot\..\src\cascadia\TerminalSettingsEditor\Generated Files\"
|
|
)
|
|
|
|
# Prohibited UIDs (exact match, case-insensitive by default)
|
|
$ProhibitedUids = @(
|
|
"Extensions_Scope",
|
|
"Profile_MissingFontFaces",
|
|
"Profile_ProportionalFontFaces",
|
|
"ColorScheme_InboxSchemeDuplicate",
|
|
"ColorScheme_ColorsHeader",
|
|
"ColorScheme_Rename"
|
|
)
|
|
|
|
# Prohibited XAML files (already limited to Page root elements)
|
|
$ProhibitedXamlFiles = @(
|
|
"AISettings.xaml",
|
|
"Profiles_Base_Orphaned.xaml",
|
|
"EditAction.xaml",
|
|
"MainPage.xaml"
|
|
)
|
|
|
|
# Grouped metadata for each page class
|
|
$ClassMap = @{
|
|
"Microsoft::Terminal::Settings::Editor::Launch" = @{
|
|
ResourceName = "Nav_Launch/Content"
|
|
NavigationParam = "Launch_Nav"
|
|
SubPage = "BreadcrumbSubPage::None"
|
|
}
|
|
"Microsoft::Terminal::Settings::Editor::Interaction" = @{
|
|
ResourceName = "Nav_Interaction/Content"
|
|
NavigationParam = "Interaction_Nav"
|
|
SubPage = "BreadcrumbSubPage::None"
|
|
}
|
|
"Microsoft::Terminal::Settings::Editor::GlobalAppearance" = @{
|
|
ResourceName = "Nav_Appearance/Content"
|
|
NavigationParam = "GlobalAppearance_Nav"
|
|
SubPage = "BreadcrumbSubPage::None"
|
|
}
|
|
"Microsoft::Terminal::Settings::Editor::ColorSchemes" = @{
|
|
ResourceName = "Nav_ColorSchemes/Content"
|
|
NavigationParam = "ColorSchemes_Nav"
|
|
SubPage = "BreadcrumbSubPage::None"
|
|
}
|
|
"Microsoft::Terminal::Settings::Editor::Rendering" = @{
|
|
ResourceName = "Nav_Rendering/Content"
|
|
NavigationParam = "Rendering_Nav"
|
|
SubPage = "BreadcrumbSubPage::None"
|
|
}
|
|
"Microsoft::Terminal::Settings::Editor::Compatibility" = @{
|
|
ResourceName = "Nav_Compatibility/Content"
|
|
NavigationParam = "Compatibility_Nav"
|
|
SubPage = "BreadcrumbSubPage::None"
|
|
}
|
|
"Microsoft::Terminal::Settings::Editor::Actions" = @{
|
|
ResourceName = "Nav_Actions/Content"
|
|
NavigationParam = "Actions_Nav"
|
|
SubPage = "BreadcrumbSubPage::None"
|
|
}
|
|
"Microsoft::Terminal::Settings::Editor::NewTabMenu" = @{
|
|
ResourceName = "Nav_NewTabMenu/Content" # [Folders] Replaced with folder name
|
|
NavigationParam = "NewTabMenu_Nav" # [Folders] Replaced with folder VM
|
|
SubPage = "BreadcrumbSubPage::None" # [Folders] Replaced with BreadcrumbSubPage::NewTabMenu_Folder
|
|
}
|
|
"Microsoft::Terminal::Settings::Editor::Extensions" = @{
|
|
ResourceName = "Nav_Extensions/Content" # [Extension] Replaced with extension name
|
|
NavigationParam = "Extensions_Nav" # [Extension] Replaced with extension VM
|
|
SubPage = "BreadcrumbSubPage::None" # [Extension] Replaced with BreadcrumbSubPage::Extensions_Extension
|
|
}
|
|
"Microsoft::Terminal::Settings::Editor::Profiles_Base" = @{
|
|
ResourceName = "Nav_ProfileDefaults/Content"
|
|
NavigationParam = "GlobalProfile_Nav"
|
|
SubPage = "BreadcrumbSubPage::None"
|
|
}
|
|
"Microsoft::Terminal::Settings::Editor::Profiles_Appearance" = @{
|
|
ResourceName = "Nav_ProfileDefaults/Content"
|
|
NavigationParam = "GlobalProfile_Nav"
|
|
SubPage = "BreadcrumbSubPage::Profile_Appearance"
|
|
}
|
|
"Microsoft::Terminal::Settings::Editor::Profiles_Terminal" = @{
|
|
ResourceName = "Nav_ProfileDefaults/Content"
|
|
NavigationParam = "GlobalProfile_Nav"
|
|
SubPage = "BreadcrumbSubPage::Profile_Terminal"
|
|
}
|
|
"Microsoft::Terminal::Settings::Editor::Profiles_Advanced" = @{
|
|
ResourceName = "Nav_ProfileDefaults/Content"
|
|
NavigationParam = "GlobalProfile_Nav"
|
|
SubPage = "BreadcrumbSubPage::Profile_Advanced"
|
|
}
|
|
"Microsoft::Terminal::Settings::Editor::AddProfile" = @{
|
|
ResourceName = "Nav_AddNewProfile/Content"
|
|
NavigationParam = "AddProfile"
|
|
SubPage = "BreadcrumbSubPage::None"
|
|
}
|
|
}
|
|
|
|
function IsProfileSubPage($pageClass)
|
|
{
|
|
return $pageClass -match "Editor::Profiles_Appearance" -or
|
|
$pageClass -match "Editor::Profiles_Terminal" -or
|
|
$pageClass -match "Editor::Profiles_Advanced"
|
|
}
|
|
|
|
if (-not (Test-Path $SourceDir)) { throw "SourceDir not found: $SourceDir" }
|
|
if (-not (Test-Path $OutputDir)) { New-Item -ItemType Directory -Path $OutputDir | Out-Null }
|
|
|
|
$entries = @()
|
|
foreach ($xamlFile in Get-ChildItem -Path $SourceDir -Filter *.xaml)
|
|
{
|
|
# Skip whole file if prohibited
|
|
$filename = $xamlFile.Name
|
|
if ($ProhibitedXamlFiles -contains $filename)
|
|
{
|
|
continue
|
|
}
|
|
|
|
# Load XAML and namespace manager
|
|
[xml]$xml = Get-Content -LiteralPath $xamlFile.FullName
|
|
$xm = New-Object System.Xml.XmlNamespaceManager($xml.NameTable)
|
|
$xm.AddNamespace("local", "using:Microsoft.Terminal.Settings.Editor")
|
|
$xm.AddNamespace("x", "http://schemas.microsoft.com/winfx/2006/xaml")
|
|
|
|
if ($xml.DocumentElement.LocalName -ne "Page" -and $filename -ne "Appearances.xaml")
|
|
{
|
|
# Only allow xaml files for Page elements (or Appearances.xaml)
|
|
continue
|
|
}
|
|
|
|
# Extract Page x:Class
|
|
# Appearances.xaml: UserControl hosted in Profiles_Appearance.xaml
|
|
$pageClass = $filename -eq "Appearances.xaml" ?
|
|
"Microsoft::Terminal::Settings::Editor::Profiles_Appearance" :
|
|
$xml.DocumentElement.SelectSingleNode("@x:Class", $xm).Value
|
|
|
|
# Convert XAML namespace dots to C++ scope operators
|
|
$pageClass = ($pageClass -replace "\.", "::")
|
|
if ($ClassMap.ContainsKey(($pageClass)) -and -not (IsProfileSubPage $pageClass))
|
|
{
|
|
$entries += [pscustomobject]@{
|
|
ResourceName = $ClassMap[$pageClass].ResourceName
|
|
ParentPage = $pageClass
|
|
NavigationParam = $ClassMap[$pageClass].NavigationParam
|
|
SubPage = $ClassMap[$pageClass].SubPage
|
|
ElementName = $null # No specific element to navigate to, for the page itself
|
|
File = $filename
|
|
}
|
|
}
|
|
elseif ($pageClass -notmatch "Editor::EditColorScheme" -and -not (IsProfileSubPage $pageClass))
|
|
{
|
|
# Special case: EditColorScheme is only valid if a color scheme is associated,
|
|
# so don't register it in ClassMap and don't skip over it here.
|
|
Write-Warning "No class map entry for page class $pageClass (file: $filename). Skipping automatic index generation for this page."
|
|
continue
|
|
}
|
|
|
|
# Manually register special entries
|
|
if ($filename -eq "ColorSchemes.xaml")
|
|
{
|
|
# "add new" button
|
|
$entries += [pscustomobject]@{
|
|
ResourceName = "ColorScheme_AddNewButton/Text"
|
|
ParentPage = $pageClass
|
|
NavigationParam = $ClassMap[$pageClass].NavigationParam
|
|
SubPage = $ClassMap[$pageClass].SubPage
|
|
ElementName = "AddNewButton"
|
|
File = $filename
|
|
}
|
|
}
|
|
elseif ($filename -eq "AddProfile.xaml")
|
|
{
|
|
# "add new" button
|
|
$entries += [pscustomobject]@{
|
|
ResourceName = "AddProfile_AddNewTextBlock/Text"
|
|
ParentPage = $pageClass
|
|
NavigationParam = $ClassMap[$pageClass].NavigationParam
|
|
SubPage = $ClassMap[$pageClass].SubPage
|
|
ElementName = "AddNewButton"
|
|
File = $filename
|
|
}
|
|
}
|
|
|
|
# Iterate over all local:SettingContainer nodes
|
|
foreach ($settingContainer in $xml.SelectNodes("//local:SettingContainer", $xm))
|
|
{
|
|
# Extract Uid
|
|
if ($null -eq $settingContainer.Uid)
|
|
{
|
|
Write-Warning "No x:Uid found for a SettingContainer in file $filename. Skipping entry."
|
|
continue
|
|
}
|
|
elseif ($ProhibitedUids -contains $settingContainer.Uid)
|
|
{
|
|
continue
|
|
}
|
|
|
|
# Extract Name via GetAttribute to avoid PowerShell's XML integration
|
|
# returning the element name (e.g. "local:SettingContainer") when x:Name is absent.
|
|
$name = $settingContainer.GetAttribute("x:Name")
|
|
if ([string]::IsNullOrEmpty($name))
|
|
{
|
|
$name = ""
|
|
}
|
|
if ($filename -eq "Appearances.xaml")
|
|
{
|
|
# Profile.Appearance settings need a special prefix for the ElementName.
|
|
# This allows us to bring the element into view at runtime.
|
|
$name = "App." + $name
|
|
}
|
|
|
|
# Deduce NavigationParam and SubPage
|
|
# includeInBuildIndex: include the entry in the build-time index (no special param at runtime)
|
|
# includeInPartialIndex: include the entry in the partial index, where the NavigationParam is the view model at runtime (i.e. profile vs profile defaults)
|
|
$includeInBuildIndex = $true
|
|
$includeInPartialIndex = $false
|
|
$navigationParam = $ClassMap[$pageClass].NavigationParam
|
|
$subPage = $ClassMap[$pageClass].SubPage ?? "BreadcrumbSubPage::None"
|
|
if ($pageClass -match "Editor::NewTabMenu")
|
|
{
|
|
if ($settingContainer.Uid -match "NewTabMenu_CurrentFolder")
|
|
{
|
|
$navigationParam = $null # VM param at runtime
|
|
$subPage = "BreadcrumbSubPage::NewTabMenu_Folder"
|
|
$includeInBuildIndex = $false
|
|
$includeInPartialIndex = $true
|
|
}
|
|
else
|
|
{
|
|
$includeInPartialIndex = $true
|
|
}
|
|
}
|
|
elseif ($pageClass -match "Editor::Profiles_Base" -or (IsProfileSubPage $pageClass))
|
|
{
|
|
$includeInBuildIndex = !($name -eq "Name" -or $name -eq "Commandline")
|
|
$includeInPartialIndex = $true
|
|
}
|
|
elseif ($pageClass -match "Editor::EditColorScheme")
|
|
{
|
|
$subPage = "BreadcrumbSubPage::ColorSchemes_Edit"
|
|
$includeInBuildIndex = $false
|
|
$includeInPartialIndex = $true
|
|
}
|
|
|
|
if ($includeInBuildIndex)
|
|
{
|
|
$entries += [pscustomobject]@{
|
|
ResourceName = "$($settingContainer.Uid)/Header"
|
|
ParentPage = $pageClass
|
|
NavigationParam = $navigationParam
|
|
SubPage = $subPage
|
|
ElementName = $name
|
|
File = $filename
|
|
}
|
|
}
|
|
|
|
if ($includeInPartialIndex)
|
|
{
|
|
$entries += [pscustomobject]@{
|
|
ResourceName = "$($settingContainer.Uid)/Header"
|
|
ParentPage = $pageClass
|
|
NavigationParam = $null # VM param at runtime
|
|
SubPage = $pageClass -match "Editor::NewTabMenu" ? "BreadcrumbSubPage::NewTabMenu_Folder" : $subPage
|
|
ElementName = $name
|
|
File = $filename
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function FormatEntry($e)
|
|
{
|
|
$formattedResourceName = 'USES_RESOURCE(L"{0}")' -f $e.ResourceName
|
|
$formattedNavigationParam = 'L"{0}"' -f $e.NavigationParam # null Navigation param resolves to empty string
|
|
$formattedElementName = 'L"{0}"' -f $e.ElementName
|
|
|
|
return " IndexEntry{{ {0}, {1}, {2}, {3} }}, // {4}" -f ($formattedResourceName, $formattedNavigationParam, $e.SubPage, $formattedElementName, $e.File)
|
|
}
|
|
|
|
function FormatEntries($es) {
|
|
return ($es | ForEach-Object { FormatEntry $_ }) -join "`r`n"
|
|
}
|
|
|
|
# Sort and remove duplicates
|
|
$entries = $entries | Sort-Object ResourceName, ParentPage, NavigationParam, SubPage, ElementName, File -Unique
|
|
|
|
$buildTimeEntries = @()
|
|
$profileEntries = @()
|
|
$schemeEntries = @()
|
|
$ntmEntries = @()
|
|
foreach ($e in $entries)
|
|
{
|
|
if ($null -eq $e.NavigationParam -and $e.ParentPage -match "Profiles_")
|
|
{
|
|
$profileEntries += $e
|
|
}
|
|
elseif ($e.SubPage -eq "BreadcrumbSubPage::ColorSchemes_Edit")
|
|
{
|
|
$schemeEntries += $e
|
|
}
|
|
elseif ($e.SubPage -eq "BreadcrumbSubPage::NewTabMenu_Folder")
|
|
{
|
|
$ntmEntries += $e
|
|
}
|
|
else
|
|
{
|
|
$buildTimeEntries += $e
|
|
}
|
|
}
|
|
|
|
$headerPath = Join-Path $OutputDir "GeneratedSettingsIndex.g.h"
|
|
$cppPath = Join-Path $OutputDir "GeneratedSettingsIndex.g.cpp"
|
|
|
|
$header = @"
|
|
/*++
|
|
Copyright (c) Microsoft Corporation
|
|
Licensed under the MIT license.
|
|
--*/
|
|
// This file is automatically generated by tools\GenerateSettingsIndex.ps1. Changes to this file may be overwritten.
|
|
#pragma once
|
|
#include <winrt/Windows.UI.Xaml.Interop.h>
|
|
|
|
namespace winrt::Microsoft::Terminal::Settings::Editor::implementation
|
|
{
|
|
struct IndexEntry
|
|
{
|
|
// Resource name of the SettingContainer's Header (i.e. "Globals_DefaultProfile/Header")
|
|
// NOTE: wrapped in USES_RESOURCE() to take advantage of compile-time resource name validation in ResourceLoader
|
|
wil::zwstring_view ResourceName;
|
|
|
|
// Navigation argument
|
|
// - the tag used to identify the page to navigate to (i.e. "Launch_Nav")
|
|
// - empty if the NavigationArg is meant to be a view model object at runtime (i.e. profile, ntm folder, etc.)
|
|
wil::zwstring_view NavigationArgTag;
|
|
|
|
// SubPage to navigate to, for pages with multiple subpages (i.e. Profiles, New Tab Menu)
|
|
BreadcrumbSubPage SubPage;
|
|
|
|
// x:Name of the SettingContainer to navigate to on the page (i.e. "DefaultProfile")
|
|
wil::zwstring_view ElementName;
|
|
};
|
|
|
|
const std::array<IndexEntry, $($buildTimeEntries.Count)>& LoadBuildTimeIndex();
|
|
const std::array<IndexEntry, $($profileEntries.Count)>& LoadProfileIndex();
|
|
const std::array<IndexEntry, $($ntmEntries.Count)>& LoadNTMFolderIndex();
|
|
const std::array<IndexEntry, $($schemeEntries.Count)>& LoadColorSchemeIndex();
|
|
|
|
const IndexEntry& PartialProfileIndexEntry();
|
|
const IndexEntry& PartialNTMFolderIndexEntry();
|
|
const IndexEntry& PartialColorSchemeIndexEntry();
|
|
const IndexEntry& PartialExtensionIndexEntry();
|
|
const IndexEntry& PartialActionIndexEntry();
|
|
}
|
|
"@
|
|
|
|
$cpp = @"
|
|
// Copyright (c) Microsoft Corporation.
|
|
// Licensed under the MIT license.
|
|
// This file is automatically generated by tools\GenerateSettingsIndex.ps1. Changes to this file may be overwritten.
|
|
|
|
#include "pch.h"
|
|
#include <winrt/Microsoft.Terminal.Settings.Editor.h>
|
|
#include "GeneratedSettingsIndex.g.h"
|
|
#include <LibraryResources.h>
|
|
|
|
// In Debug builds, USES_RESOURCE() expands to a lambda (for resource validation),
|
|
// which prevents constexpr evaluation. In Release it's a no-op identity macro.
|
|
#ifdef _DEBUG
|
|
#define STATIC_INDEX_QUALIFIER static const
|
|
#else
|
|
#define STATIC_INDEX_QUALIFIER static constexpr
|
|
#endif
|
|
|
|
namespace winrt::Microsoft::Terminal::Settings::Editor::implementation
|
|
{
|
|
const std::array<IndexEntry, $($buildTimeEntries.Count)>& LoadBuildTimeIndex()
|
|
{
|
|
STATIC_INDEX_QUALIFIER std::array entries =
|
|
{
|
|
$(FormatEntries $buildTimeEntries)
|
|
};
|
|
return entries;
|
|
}
|
|
|
|
const std::array<IndexEntry, $($profileEntries.Count)>& LoadProfileIndex()
|
|
{
|
|
STATIC_INDEX_QUALIFIER std::array entries =
|
|
{
|
|
$(FormatEntries $profileEntries)
|
|
};
|
|
return entries;
|
|
}
|
|
|
|
const std::array<IndexEntry, $($ntmEntries.Count)>& LoadNTMFolderIndex()
|
|
{
|
|
STATIC_INDEX_QUALIFIER std::array entries =
|
|
{
|
|
$(FormatEntries $ntmEntries)
|
|
};
|
|
return entries;
|
|
}
|
|
|
|
const std::array<IndexEntry, $($schemeEntries.Count)>& LoadColorSchemeIndex()
|
|
{
|
|
STATIC_INDEX_QUALIFIER std::array entries =
|
|
{
|
|
$(FormatEntries $schemeEntries)
|
|
};
|
|
return entries;
|
|
}
|
|
|
|
const IndexEntry& PartialProfileIndexEntry()
|
|
{
|
|
static constexpr IndexEntry entry{ .SubPage = BreadcrumbSubPage::None };
|
|
return entry;
|
|
}
|
|
|
|
const IndexEntry& PartialNTMFolderIndexEntry()
|
|
{
|
|
static constexpr IndexEntry entry{ .SubPage = BreadcrumbSubPage::NewTabMenu_Folder };
|
|
return entry;
|
|
}
|
|
|
|
const IndexEntry& PartialColorSchemeIndexEntry()
|
|
{
|
|
static constexpr IndexEntry entry{ .SubPage = BreadcrumbSubPage::ColorSchemes_Edit };
|
|
return entry;
|
|
}
|
|
|
|
const IndexEntry& PartialExtensionIndexEntry()
|
|
{
|
|
static constexpr IndexEntry entry{ .SubPage = BreadcrumbSubPage::Extensions_Extension };
|
|
return entry;
|
|
}
|
|
|
|
const IndexEntry& PartialActionIndexEntry()
|
|
{
|
|
static constexpr IndexEntry entry{ .SubPage = BreadcrumbSubPage::Actions_Edit };
|
|
return entry;
|
|
}
|
|
}
|
|
"@
|
|
|
|
Set-Content -LiteralPath $headerPath -Value $header -NoNewline
|
|
Set-Content -LiteralPath $cppPath -Value $cpp -NoNewline
|
|
|
|
Write-Host "Generated:"
|
|
Write-Host " $headerPath"
|
|
Write-Host " $cppPath" |