diff --git a/src/ThisAssembly.Constants/ConstantsGenerator.cs b/src/ThisAssembly.Constants/ConstantsGenerator.cs
index 1c15ed7c..252a8964 100644
--- a/src/ThisAssembly.Constants/ConstantsGenerator.cs
+++ b/src/ThisAssembly.Constants/ConstantsGenerator.cs
@@ -3,6 +3,7 @@
using System.IO;
using System.Linq;
using System.Text;
+using System.Text.RegularExpressions;
using Devlooped.Sponsors;
using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.CSharp;
@@ -16,6 +17,8 @@ namespace ThisAssembly;
[Generator(LanguageNames.CSharp)]
public class ConstantsGenerator : IIncrementalGenerator
{
+ static readonly Regex SeeExpr = new("", RegexOptions.Compiled);
+
public void Initialize(IncrementalGeneratorInitializationContext context)
{
var files = context.AdditionalTextsProvider
@@ -121,12 +124,16 @@ void GenerateConstant(SourceProductionContext spc,
// structures via functions.
if (parse.Language == LanguageNames.CSharp)
{
- output = SyntaxFactory.ParseCompilationUnit(output, options: cs)
+ output = SeeExpr.Replace(SyntaxFactory
+ .ParseCompilationUnit(output, options: cs)
.NormalizeWhitespace()
.GetText()
- .ToString();
+ .ToString(),
+ $"");
}
spc.AddSource($"{root}.{name}.g.cs", SourceText.From(output, Encoding.UTF8));
}
+
+
}
diff --git a/src/ThisAssembly.Resources/CSharp.sbntxt b/src/ThisAssembly.Resources/CSharp.sbntxt
index 419a262c..82b9c55d 100644
--- a/src/ThisAssembly.Resources/CSharp.sbntxt
+++ b/src/ThisAssembly.Resources/CSharp.sbntxt
@@ -57,5 +57,16 @@ namespace {{ Namespace }};
///
/// Provides access to assembly resources.
///
+ {{~ if Remarks ~}}
+ {{ Remarks }}
+ ///
+ {{~ end ~}}
+ {{~ if Warn ~}}
+ [Obsolete("{{ Warn }}", false
+ #if NET6_0_OR_GREATER
+ , UrlFormat = "{{ Url }}"
+ #endif
+ )]
+ {{~ end ~}}
{{ render RootArea }}
}
diff --git a/src/ThisAssembly.Resources/Model.cs b/src/ThisAssembly.Resources/Model.cs
index d2e69139..4652e2af 100644
--- a/src/ThisAssembly.Resources/Model.cs
+++ b/src/ThisAssembly.Resources/Model.cs
@@ -9,6 +9,9 @@ namespace ThisAssembly;
[DebuggerDisplay("Values = {RootArea.Values.Count}")]
record Model(Area RootArea, string? Namespace, bool IsPublic)
{
+ public string? Warn { get; set; }
+ public string? Remarks { get; set; }
+ public string Url => Devlooped.Sponsors.SponsorLink.Funding.HelpUrl;
public string Version => Assembly.GetExecutingAssembly().GetName().Version.ToString(3);
public string Visibility => IsPublic ? "public " : "";
}
diff --git a/src/ThisAssembly.Resources/ResourcesGenerator.cs b/src/ThisAssembly.Resources/ResourcesGenerator.cs
index 8e2ff04a..7b5ce772 100644
--- a/src/ThisAssembly.Resources/ResourcesGenerator.cs
+++ b/src/ThisAssembly.Resources/ResourcesGenerator.cs
@@ -1,17 +1,24 @@
using System;
using System.Collections.Immutable;
+using System.Globalization;
using System.IO;
using System.Linq;
using System.Text;
+using System.Text.RegularExpressions;
+using Devlooped.Sponsors;
using Microsoft.CodeAnalysis;
+using Microsoft.CodeAnalysis.CSharp;
using Microsoft.CodeAnalysis.Text;
using Scriban;
+using static Devlooped.Sponsors.SponsorLink;
namespace ThisAssembly;
[Generator(LanguageNames.CSharp)]
public class ResourcesGenerator : IIncrementalGenerator
{
+ static readonly Regex SeeExpr = new("", RegexOptions.Compiled);
+
public void Initialize(IncrementalGeneratorInitializationContext context)
{
context.RegisterPostInitializationOutput(
@@ -52,16 +59,19 @@ public void Initialize(IncrementalGeneratorInitializationContext context)
c.GlobalOptions.TryGetValue("build_property.ThisAssemblyVisibility", out var visibility) && !string.IsNullOrEmpty(visibility) ? visibility : null
));
+ // this is required to ensure status is registered properly independently of analyzer runs.
+ var options = context.GetStatusOptions();
+
context.RegisterSourceOutput(
- files.Combine(right),
+ files.Combine(right).Combine(options.Combine(context.ParseOptionsProvider)),
GenerateSource);
}
static void GenerateSource(SourceProductionContext spc,
- ((ImmutableArray<(string resourceName, string? kind, string? comment)> files,
- ImmutableArray extensions), (string? ns, string? visibility)) args)
+ (((ImmutableArray<(string resourceName, string? kind, string? comment)> files,
+ ImmutableArray extensions), (string? ns, string? visibility)), (StatusOptions options, ParseOptions parse)) args)
{
- var ((files, extensions), (ns, visibility)) = args;
+ var (((files, extensions), (ns, visibility)), (options, parse)) = args;
var file = "CSharp.sbntxt";
var template = Template.Parse(EmbeddedResource.GetContent(file), file);
@@ -70,6 +80,23 @@ static void GenerateSource(SourceProductionContext spc,
.GroupBy(f => Path.Combine(
Path.GetDirectoryName(f.resourceName),
Path.GetFileNameWithoutExtension(f.resourceName)));
+
+ string? warn = default;
+ string? remarks = default;
+ if (IsEditor)
+ {
+ var status = Diagnostics.GetOrSetStatus(options);
+ if (status == SponsorStatus.Unknown || status == SponsorStatus.Expired)
+ {
+ warn = string.Format(CultureInfo.CurrentCulture, Resources.Editor_Disabled, Funding.Product, Funding.HelpUrl);
+ remarks = Resources.Editor_DisabledRemarks;
+ }
+ else if (status == SponsorStatus.Grace && Diagnostics.TryGet() is { } grace && grace.Properties.TryGetValue(nameof(SponsorStatus.Grace), out var days))
+ {
+ remarks = string.Format(CultureInfo.CurrentCulture, Resources.Editor_GraceRemarks, days);
+ }
+ }
+
foreach (var group in groupsWithoutExtensions)
{
var basePath = group.Key;
@@ -90,16 +117,19 @@ static void GenerateSource(SourceProductionContext spc,
.ToList();
var root = Area.Load(basePath, resources);
- var model = new Model(root, ns, "public".Equals(visibility, StringComparison.OrdinalIgnoreCase));
-
+ var model = new Model(root, ns, "public".Equals(visibility, StringComparison.OrdinalIgnoreCase))
+ {
+ Warn = warn,
+ Remarks = remarks,
+ };
var output = template.Render(model, member => member.Name);
- // Apply formatting since indenting isn't that nice in Scriban when rendering nested
- // structures via functions.
- output = Microsoft.CodeAnalysis.CSharp.SyntaxFactory.ParseCompilationUnit(output)
+ output = SeeExpr.Replace(SyntaxFactory
+ .ParseCompilationUnit(output, options: parse as CSharpParseOptions)
.NormalizeWhitespace()
.GetText()
- .ToString();
+ .ToString(),
+ $"");
spc.AddSource(
$"{basePath.Replace('\\', '.').Replace('/', '.')}.g.cs",
diff --git a/src/ThisAssembly.Resources/ThisAssembly.Resources.props b/src/ThisAssembly.Resources/ThisAssembly.Resources.props
index 16795a20..ca8ce68b 100644
--- a/src/ThisAssembly.Resources/ThisAssembly.Resources.props
+++ b/src/ThisAssembly.Resources/ThisAssembly.Resources.props
@@ -1,7 +1,7 @@
- .txt|.cs|.sql|.json|.md
+ .txt|.cs|.sql|.json|.md|.resx
diff --git a/src/ThisAssembly.Strings/StringsGenerator.cs b/src/ThisAssembly.Strings/StringsGenerator.cs
index 5bf298e3..7c4fdebb 100644
--- a/src/ThisAssembly.Strings/StringsGenerator.cs
+++ b/src/ThisAssembly.Strings/StringsGenerator.cs
@@ -2,8 +2,8 @@
using System.Globalization;
using System.IO;
using System.Linq;
-using System.Resources;
using System.Text;
+using System.Text.RegularExpressions;
using Devlooped.Sponsors;
using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.CSharp;
@@ -16,6 +16,8 @@ namespace ThisAssembly;
[Generator(LanguageNames.CSharp)]
public class StringsGenerator : IIncrementalGenerator
{
+ static readonly Regex SeeExpr = new("", RegexOptions.Compiled);
+
public void Initialize(IncrementalGeneratorInitializationContext context)
{
// Read the ThisAssemblyNamespace property or default to null
@@ -83,10 +85,12 @@ static void GenerateSource(SourceProductionContext spc,
var output = template.Render(model, member => member.Name);
- output = SyntaxFactory.ParseCompilationUnit(output, options: parse as CSharpParseOptions)
+ output = SeeExpr.Replace(SyntaxFactory
+ .ParseCompilationUnit(output, options: parse as CSharpParseOptions)
.NormalizeWhitespace()
.GetText()
- .ToString();
+ .ToString(),
+ $"");
spc.AddSource(resourceName, SourceText.From(output, Encoding.UTF8));
}