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

[DRAFT] DependencyPropertyGenerator #617

Closed
wants to merge 128 commits into from
Closed
Changes from 1 commit
Commits
Show all changes
128 commits
Select commit Hold shift + click to select a range
054506c
Generated DependencyPropertyGenerator component from template
Arlodotexe Dec 2, 2024
4ed6172
Port files from private repo
Sergio0694 Dec 2, 2024
794d0c3
Cleanup and merge ported csproj, rename files
Arlodotexe Dec 2, 2024
bd3f0c5
Fix default namespace and package name for DependencyPropertyGenerato…
Arlodotexe Dec 2, 2024
28b5c83
Disable globalusings in DependencyPropertyGenerator
Arlodotexe Dec 2, 2024
8e7694f
Move files, fix folder name
Arlodotexe Dec 3, 2024
54f0445
Bring back embedded resources
Sergio0694 Dec 3, 2024
9aec601
Fix folder name typo
Arlodotexe Dec 3, 2024
f472ab0
Fix project references and update embedded resource paths in Dependen…
Arlodotexe Dec 3, 2024
a559c58
Fix wrong paths in test project
Sergio0694 Dec 3, 2024
bf62d4d
Fix more wrong paths
Sergio0694 Dec 3, 2024
eb854c9
Fix some warnings in the test project
Sergio0694 Dec 3, 2024
df93079
Create .gitattributes
Sergio0694 Dec 3, 2024
03f041e
Fix newlines to LF
Sergio0694 Dec 3, 2024
d503756
Use 'WellKnownTypeNames' in all analyzers
Sergio0694 Dec 3, 2024
8846a70
Remove global usings
Sergio0694 Dec 3, 2024
51d44bd
Finish 'InvalidPropertyDefaultValueTypeAnalyzer'
Sergio0694 Dec 3, 2024
6b2d0cb
Add unit tests
Sergio0694 Dec 3, 2024
6c1ceb7
Consolidate GlobalUsings imports around tooling
Arlodotexe Dec 3, 2024
d9bec94
Fixed namespace errors when running under Uno
Arlodotexe Dec 3, 2024
291f7fa
Update tooling submodule
Arlodotexe Dec 3, 2024
9637c3a
Remove LangVersion setting from DependencyPropertyGenerator project
Arlodotexe Dec 3, 2024
9f0809d
Ran XAML styler
Arlodotexe Dec 3, 2024
8436bde
Remove empty samples
Arlodotexe Dec 3, 2024
1babcec
Remove empty component doc to fix CI error
Arlodotexe Dec 3, 2024
c2351ac
Remove unused test files from DependencyPropertyGenerator project
Arlodotexe Dec 3, 2024
471c161
Update .NET version to 9.0 in Dockerfile, devcontainer.json, build.ym…
Arlodotexe Dec 5, 2024
9657827
Fixed compilation conditionals
Arlodotexe Dec 5, 2024
10499ac
Temporarily limit WinUI and multitarget options in build matrix for U…
Arlodotexe Dec 5, 2024
da2b2c3
Temporarily disable wasm-linux check, only build DependencyPropertyGe…
Arlodotexe Dec 5, 2024
cf30a64
Adjust CI to only build DependencyPropertyGenerator
Arlodotexe Dec 5, 2024
ec018ef
Enable WinUI 3, fix GenerateSingleSampleHeads script invocation
Arlodotexe Dec 5, 2024
c937c4c
Switch generator to WinAppSDK for now
Sergio0694 Dec 5, 2024
9acc0ce
Temp: Refactor build workflow to use Build-Toolkit-Components script …
Arlodotexe Dec 5, 2024
05edd93
Fix condition for local caching in generator
Sergio0694 Dec 5, 2024
c7ab831
Define 'DependencyPropertyGeneratorUseWindowsUIXaml'
Sergio0694 Dec 5, 2024
aff5de9
Generalize .dll reference check
Sergio0694 Dec 5, 2024
9ee417e
Add 'ForAttributeWithMetadataNameAndOptions'
Sergio0694 Dec 5, 2024
061f74e
Add 'AnalyzerConfigOptionsExtensions'
Sergio0694 Dec 5, 2024
6af9be8
Enable XAML option for generators/analyzers
Sergio0694 Dec 5, 2024
c100c0e
Enable Windows.UI.Xaml for legacy UWP
Sergio0694 Dec 5, 2024
9439f37
Add 'SyntaxKind' extensions
Sergio0694 Dec 6, 2024
eb41e28
Add support for more modifiers, bug fixes
Sergio0694 Dec 6, 2024
1b40caf
Add unit tests for more modifiers
Sergio0694 Dec 6, 2024
e152803
Add diagnostic and test for pointer types
Sergio0694 Dec 6, 2024
95590f0
Add 'DefaultValueCallback' property
Sergio0694 Dec 6, 2024
22ab868
Add initial support for default value callbacks
Sergio0694 Dec 6, 2024
c053699
Fix value callbacks, add unit tests
Sergio0694 Dec 7, 2024
b0e178e
Add '[DisallowNull]' to 'DefaultValueCallback'
Sergio0694 Dec 7, 2024
b4bf0a3
Add 'InvalidPropertyDefaultValueCallbackTypeAnalyzer'
Sergio0694 Dec 7, 2024
ba00de9
Add unit tests for new analyzer
Sergio0694 Dec 7, 2024
8f971c9
Fix codegen for all default value callback cases
Sergio0694 Dec 7, 2024
aa12670
Add 'PropertyDeclarationWithPropertyNameSuffixAnalyzer'
Sergio0694 Dec 7, 2024
e988aab
Fix two nullability warnings
Sergio0694 Dec 7, 2024
4a42758
Improve formatting for property initialization
Sergio0694 Dec 7, 2024
d40bee5
Bug fixes to embedded mode
Sergio0694 Dec 9, 2024
b48e37f
Update components/DependencyPropertyGenerator/CommunityToolkit.Depend…
Arlodotexe Dec 9, 2024
a943c7b
Update components/DependencyPropertyGenerator/CommunityToolkit.Depend…
Arlodotexe Dec 9, 2024
42d3211
Fix typos
Sergio0694 Dec 9, 2024
e481f48
Add draft 'UseGeneratedDependencyPropertyOnManualPropertyAnalyzer'
Sergio0694 Dec 12, 2024
ef70aab
Pass the location of the target field
Sergio0694 Dec 12, 2024
08707e6
Add empty 'CodeFixers' project
Sergio0694 Dec 12, 2024
2389d33
Bump 'Microsoft.CodeAnalysis.CSharp' to latest
Sergio0694 Dec 12, 2024
0281e36
Pack code fixers into NuGet package
Sergio0694 Dec 12, 2024
a032d85
Add 'InternalsVisibleTo' for code fixers
Sergio0694 Dec 12, 2024
8524021
Add draft 'UseGeneratedDependencyPropertyOnManualPropertyCodeFixer'
Sergio0694 Dec 12, 2024
df50c2f
Add 'CSharpCodeFixTest<,>' type
Sergio0694 Dec 12, 2024
7f57e6c
Add basic code fixer test
Sergio0694 Dec 12, 2024
866492b
Optimize registration for default values
Sergio0694 Dec 12, 2024
26fdeed
Handle more default properties in analyzer
Sergio0694 Dec 12, 2024
bebdcac
Handle even more default properties in analyzer
Sergio0694 Dec 12, 2024
be37f5f
Add more generator unit tests
Sergio0694 Dec 13, 2024
26d8f03
Handle supported default value types
Sergio0694 Dec 13, 2024
a3e4c02
Remove unnecessary using directive
Sergio0694 Dec 13, 2024
d8aee78
Fix handling of numerics, add more projected types
Sergio0694 Dec 13, 2024
a7c5130
Handle projected enums in analyzer, add tests
Sergio0694 Dec 13, 2024
3c250da
Fix some analyzer bugs, add unit tests
Sergio0694 Dec 13, 2024
49786ff
Merge branch 'main' into component/DependencyPropertyGenerator
Arlodotexe Dec 13, 2024
3241ea5
Adjust priority for enum typed constants
Sergio0694 Dec 13, 2024
7d65ea1
Fix handling of defaulted custom structs in analyzer
Sergio0694 Dec 13, 2024
e48226c
Improve formatting for known enum members
Sergio0694 Dec 13, 2024
4d085d3
Improve codegen for property changed callbacks
Sergio0694 Dec 16, 2024
bea72f9
Add incrementality tests
Sergio0694 Dec 18, 2024
0262e78
Remove unnecessary test code
Sergio0694 Dec 18, 2024
c315a6c
Fix some leftovers
Sergio0694 Dec 20, 2024
4873405
Remove unnecessary parentheses
Sergio0694 Dec 20, 2024
fc423c5
Improve code fixer for known enum members
Sergio0694 Dec 20, 2024
b65e899
Add .targets to .NET 9 folders too
Sergio0694 Dec 21, 2024
7524a0c
Simplify code fixer tests
Sergio0694 Dec 25, 2024
b85e1bc
Merge branch 'main' into component/DependencyPropertyGenerator
Sergio0694 Dec 25, 2024
45029c6
Undo temporary hacks and workarounds
Sergio0694 Dec 25, 2024
f888673
Fix tooling submodule pointer
Sergio0694 Dec 25, 2024
079da46
Only build for UWP and WindowsAppSDK
Sergio0694 Dec 25, 2024
cce95d3
Fix TFMs for UWP projects
Sergio0694 Dec 25, 2024
a4ced34
Simplify packed .targets files
Sergio0694 Dec 25, 2024
65fdb63
Disable 'UseUwpTools' property
Sergio0694 Dec 25, 2024
5a89531
Fix filenames for packaged .targets files
Sergio0694 Dec 25, 2024
ea71909
Fix typos in .targets file
Sergio0694 Dec 26, 2024
a2f339c
Fix .targets again, add basic test
Sergio0694 Dec 26, 2024
4e5bc48
Bump tooling pointer
Sergio0694 Dec 26, 2024
191e386
Bump UWP .NET 9 target SDK to 18362
Sergio0694 Dec 26, 2024
9924f77
Add more code fixer tests
Sergio0694 Dec 26, 2024
0c45212
Improve handling of known constants
Sergio0694 Dec 26, 2024
500ed36
Improve number formatting, add fixer tests
Sergio0694 Dec 26, 2024
f4ec6f5
Bump SDK of UWP on .NET 9 to 19041
Sergio0694 Dec 26, 2024
53f3195
Remove Uno workaround, fix TFMs
Sergio0694 Dec 26, 2024
f3613a3
Trivia handling
Youssef1313 Dec 26, 2024
c031808
Use the lambda overload
Youssef1313 Dec 26, 2024
9ae5fba
Tweak WinRT types matching logic
Sergio0694 Dec 27, 2024
74919d5
Fix AppServices build
Sergio0694 Dec 27, 2024
7d5c559
Add more test coverage for XML docs
Sergio0694 Dec 27, 2024
17bc997
Fix handling of EOLs in removed members
Sergio0694 Dec 27, 2024
1dadd4d
Fix EOL handling in more scenarios
Sergio0694 Dec 27, 2024
f456963
Respect accessibility of generated accessors
Sergio0694 Dec 27, 2024
70d5857
Add 'SyntaxTriviaExtensions', code tweaks
Sergio0694 Dec 27, 2024
072c750
Add 'AttributeInfo' type and associated logic
Sergio0694 Dec 27, 2024
9af0344
Gather forwarded attributes for generated fields
Sergio0694 Dec 27, 2024
e56f8ae
Fix attributes generation, add unit tests
Sergio0694 Dec 27, 2024
f75dcd5
Update analyzer for attributes, add unit tests
Sergio0694 Dec 27, 2024
e662f76
Update code fixer, add unit tests
Sergio0694 Dec 27, 2024
e617cf0
Add 'StaticAttributeListTargetOnGeneratedDependencyPropertyDeclaratio…
Sergio0694 Dec 27, 2024
2a5f0fe
Add analyzer for forwarded attributes, and tests
Sergio0694 Dec 27, 2024
81e2ad3
Fix outdated comments
Sergio0694 Dec 27, 2024
8a048a8
Strip trivia from forwarded attributes
Sergio0694 Dec 27, 2024
c36129f
Add 'CSharpSuppressorTest<TSuppressor>'
Sergio0694 Dec 27, 2024
c372d84
Fix diagnostic suppressor, add tests
Sergio0694 Dec 27, 2024
e37d7dc
Add interleaved non-fixable properties, add tests
Sergio0694 Dec 28, 2024
21f3684
Add 'partial' for nested types too, add tests
Sergio0694 Dec 28, 2024
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
Prev Previous commit
Next Next commit
Gather forwarded attributes for generated fields
  • Loading branch information
Sergio0694 committed Dec 27, 2024

Verified

This commit was created on GitHub.com and signed with GitHub’s verified signature.
commit 9af0344ea75dec821e65684c60ca139fdb5477d3
Original file line number Diff line number Diff line change
@@ -3,6 +3,7 @@
// See the LICENSE file in the project root for more information.

using System;
using System.Collections.Generic;
using System.Collections.Immutable;
using System.IO;
using System.Linq;
@@ -393,6 +394,73 @@ public static bool IsSharedPropertyChangedCallbackImplemented(IPropertySymbol pr
return false;
}

/// <summary>
/// Gathers all forwarded attributes for the generated property.
/// </summary>
///<param name="node">The input <see cref="PropertyDeclarationSyntax"/> node.</param>
/// <param name="semanticModel">The <see cref="SemanticModel"/> instance for the current run.</param>
/// <param name="forwardedAttributes">The collection of forwarded attributes to add new ones to.</param>
/// <param name="diagnostics">The current collection of gathered diagnostics.</param>
/// <param name="token">The cancellation token for the current operation.</param>
public static void GetForwardedAttributes(
PropertyDeclarationSyntax node,
SemanticModel semanticModel,
CancellationToken token,
out ImmutableArray<AttributeInfo> staticFieldAttributes)
{
using ImmutableArrayBuilder<AttributeInfo> builder = new();

// Gather explicit forwarded attributes info
foreach (AttributeListSyntax attributeList in node.AttributeLists)
{
// Only look for the 'static' attribute target, which can be used to target the generated 'DependencyProperty' static field.
// Roslyn will normally emit a 'CS0657' warning (invalid target), but that is automatically suppressed by a dedicated diagnostic
// suppressor that recognizes uses of this target specifically to support '[GeneratedDependencyProperty]'. We can't use 'field'
// as trigger, as that's used for the actual 'field' keyword, when local caching is enabled.
if (attributeList.Target?.Identifier is not SyntaxToken(SyntaxKind.StaticKeyword))
{
continue;
}

token.ThrowIfCancellationRequested();

foreach (AttributeSyntax attribute in attributeList.Attributes)
{
// Roslyn ignores attributes in an attribute list with an invalid target, so we can't get the 'AttributeData' as usual.
// To reconstruct all necessary attribute info to generate the serialized model, we use the following steps:
// - We try to get the attribute symbol from the semantic model, for the current attribute syntax. In case this is not
// available (in theory it shouldn't, but it can be), we try to get it from the candidate symbols list for the node.
// If there are no candidates or more than one, we just issue a diagnostic and stop processing the current attribute.
// The returned symbols might be method symbols (constructor attribute) so in that case we can get the declaring type.
// - We then go over each attribute argument expression and get the operation for it. This will still be available even
// though the rest of the attribute is not validated nor bound at all. From the operation we can still retrieve all
// constant values to build the 'AttributeInfo' model. After all, attributes only support constant values, 'typeof(T)'
// expressions, or arrays of either these two types, or of other arrays with the same rules, recursively.
// - From the syntax, we can also determine the identifier names for named attribute arguments, if any.
//
// There is no need to validate anything here: the attribute will be forwarded as is, and then Roslyn will validate on the
// generated property. Users will get the same validation they'd have had directly over the field. The only drawback is the
// lack of IntelliSense when constructing attributes over the field, but this is the best we can do from this end anyway.
if (!semanticModel.GetSymbolInfo(attribute, token).TryGetAttributeTypeSymbol(out INamedTypeSymbol? attributeTypeSymbol))
{
continue;
}

IEnumerable<AttributeArgumentSyntax> attributeArguments = attribute.ArgumentList?.Arguments ?? [];

// Try to extract the forwarded attribute
if (!AttributeInfo.TryCreate(attributeTypeSymbol, semanticModel, attributeArguments, token, out AttributeInfo? attributeInfo))
{
continue;
}

builder.Add(attributeInfo);
}
}

staticFieldAttributes = builder.ToImmutable();
}

/// <summary>
/// Writes all implementations of partial dependency property declarations.
/// </summary>
@@ -514,6 +582,13 @@ static string GetExpressionWithTrailingSpace(Accessibility accessibility)
/// </summary>
""", isMultiline: true);
writer.WriteGeneratedAttributes(GeneratorName, includeNonUserCodeAttributes: false);

// Write any forwarded attributes
foreach (AttributeInfo attributeInfo in propertyInfo.StaticFieldAttributes)
{
writer.WriteLine(attributeInfo.ToString());
}

writer.Write($$"""
public static readonly global::{{WellKnownTypeNames.DependencyProperty(propertyInfo.UseWindowsUIXaml)}} {{propertyInfo.PropertyName}}Property = global::{{WellKnownTypeNames.DependencyProperty(propertyInfo.UseWindowsUIXaml)}}.Register(
name: "{{propertyInfo.PropertyName}}",
Original file line number Diff line number Diff line change
@@ -121,6 +121,15 @@ public void Initialize(IncrementalGeneratorInitializationContext context)

token.ThrowIfCancellationRequested();

// Get any forwarded attributes
Execute.GetForwardedAttributes(
(PropertyDeclarationSyntax)context.TargetNode,
context.SemanticModel,
token,
out ImmutableArray<AttributeInfo> staticFieldAttributes);

token.ThrowIfCancellationRequested();

// Finally, get the hierarchy too
HierarchyInfo hierarchyInfo = HierarchyInfo.From(propertySymbol.ContainingType);

@@ -141,7 +150,8 @@ public void Initialize(IncrementalGeneratorInitializationContext context)
IsPropertyChangedCallbackImplemented: isPropertyChangedCallbackImplemented,
IsSharedPropertyChangedCallbackImplemented: isSharedPropertyChangedCallbackImplemented,
IsNet8OrGreater: isNet8OrGreater,
UseWindowsUIXaml: useWindowsUIXaml);
UseWindowsUIXaml: useWindowsUIXaml,
StaticFieldAttributes: staticFieldAttributes);
})
.WithTrackingName(WellKnownTrackingNames.Execute)
.Where(static item => item is not null)!;
Original file line number Diff line number Diff line change
@@ -25,6 +25,7 @@ namespace CommunityToolkit.GeneratedDependencyProperty.Models;
/// <param name="IsSharedPropertyChangedCallbackImplemented">Indicates whether the WinRT-based shared property changed callback is implemented.</param>
/// <param name="IsNet8OrGreater">Indicates whether the current target is .NET 8 or greater.</param>
/// <param name="UseWindowsUIXaml">Whether to use the UWP XAML or WinUI 3 XAML namespaces.</param>
/// <param name="StaticFieldAttributes">The attributes to emit on the generated static field, if any.</param>
internal sealed record DependencyPropertyInfo(
HierarchyInfo Hierarchy,
string PropertyName,
@@ -40,4 +41,5 @@ internal sealed record DependencyPropertyInfo(
bool IsPropertyChangedCallbackImplemented,
bool IsSharedPropertyChangedCallbackImplemented,
bool IsNet8OrGreater,
bool UseWindowsUIXaml);
bool UseWindowsUIXaml,
EquatableArray<AttributeInfo> StaticFieldAttributes);