Overview
Target Frameworks
Prerequisites
Installation
Usage
API
Building and Testing
Related Jering Projects
Related Concepts
Contributing
About
Jering.Web.SyntaxHighlighters.HighlightJS enables you to perform syntax highlighting from C# projects using HighlightJS. This library is built to be flexible; you can use a dependency injection (DI) based API or a static API.
Here is an example of syntax highlighting using the static API:
string code = @"public string ExampleFunction(string arg)
{
// Example comment
return arg + ""dummyString"";
}";
// Highlight code
string result = await StaticHighlightJSService.HighlightAsync(code, "csharp");
string syntaxHighlightedCode = @"<span class=""hljs-function""><span class=""hljs-keyword"">public</span> <span class=""hljs-built_in"">string</span> <span class=""hljs-title"">ExampleFunction</span>(<span class=""hljs-params""><span class=""hljs-built_in"">string</span> arg</span>)</span>
{
<span class=""hljs-comment"">// Example comment</span>
<span class=""hljs-keyword"">return</span> arg + <span class=""hljs-string"">"dummyString"</span>;
}";
// result == syntax highlighted code
Assert.Equal(syntaxHighlightedCode, result);
And here is an example of syntax highlighting using the DI based API:
string code = @"public string ExampleFunction(string arg)
{
// Example comment
return arg + ""dummyString"";
}";
// Highlight code
var services = new ServiceCollection();
services.AddHighlightJS();
ServiceProvider serviceProvider = services.BuildServiceProvider();
IHighlightJSService highlightJSService = serviceProvider.GetRequiredService<IHighlightJSService>();
string result = await highlightJSService.HighlightAsync(code, "csharp");
string syntaxHighlightedCode = @"<span class=""hljs-function""><span class=""hljs-keyword"">public</span> <span class=""hljs-built_in"">string</span> <span class=""hljs-title"">ExampleFunction</span>(<span class=""hljs-params""><span class=""hljs-built_in"">string</span> arg</span>)</span>
{
<span class=""hljs-comment"">// Example comment</span>
<span class=""hljs-keyword"">return</span> arg + <span class=""hljs-string"">"dummyString"</span>;
}";
// result == syntax highlighted code
Assert.Equal(syntaxHighlightedCode, result);
- .NET Standard 2.0
- .NET Framework 4.6.1
NodeJS must be installed and node.exe's directory must be added to the Path
environment variable.
This library has been tested with NodeJS 10.5.2 - 12.13.0.
Using Package Manager:
PM> Install-Package Jering.Web.SyntaxHighlighters.HighlightJS
Using .Net CLI:
> dotnet add package Jering.Web.SyntaxHighlighters.HighlightJS
This library uses depedency injection (DI) to facilitate extensibility and testability. You can use any DI framework that has adapters for Microsoft.Extensions.DependencyInjection. Here, we'll use the vanilla Microsoft.Extensions.DependencyInjection framework:
var services = new ServiceCollection();
services.AddHighlightJS();
ServiceProvider serviceProvider = services.BuildServiceProvider();
IHighlightJSService highlightJSService = serviceProvider.GetRequiredService<IHighlightJSService>();
IHighlightJSService
is a singleton service and IHighlightJSService
's members are thread safe.
Where possible, inject IHighlightJSService
into your types or keep a reference to a shared IHighlightJSService
instance.
Try to avoid creating multiple IHighlightJSService
instances, since each instance spawns a NodeJS process.
When you're done, you can manually dispose of an IHighlightJSService
instance by calling
highlightJSService.Dispose();
or
serviceProvider.Dispose(); // Calls Dispose on objects it has instantiated that are disposable
Dispose
kills the spawned NodeJS process.
Note that even if Dispose
isn't called manually, the service that manages the NodeJS process, INodeJSService
from Jering.Javascript.NodeJS, will kill the
NodeJS process when the application shuts down - if the application shuts down gracefully. If the application does not shutdown gracefully, the NodeJS process will kill
itself when it detects that its parent has been killed.
Essentially, manually disposing of IHighlightJSService
instances is not mandatory.
This library also provides a static API as an alternative. The StaticHighlightJSService
type wraps an IHighlightJSService
instance, exposing most of its public members statically.
Whether you use the static API or the DI based API depends on your development needs. If you are already using DI, if you want to mock
out syntax highlighting in your tests or if you want to overwrite services, use the DI based API. Otherwise,
use the static API. An example usage:
string result = await StaticHighlightJSService.HighlightAsync(code, "csharp");
The following section on using IHighlightJSService
applies to usage of StaticHighlightJSService
.
Code can be highlighted using IHighlightJSService.HighlightAsync
:
string code = @"public string ExampleFunction(string arg)
{
// Example comment
return arg + ""dummyString"";
}";
string highlightedCode = await highlightJSService.HighlightAsync(code, "csharp");
The second parameter of IHighlightJSService.HighlightAsync
must be a valid HighlightJS language alias.
Task<string> HighlightAsync(string code, string languageAlias, string classPrefix = "hljs-", CancellationToken cancellationToken = default)
Highlights code of a specified language.
code
- Type:
string
- Description: Code to highlight.
- Type:
languageAlias
- Type:
string
- Description: A HighlightJS language alias. Visit https://github.com/highlightjs/highlight.js/tree/master/src/languages for the list of valid language aliases.
- Type:
classPrefix
- Type:
string
- Description: If not null or whitespace, this string will be appended to HighlightJS classes. Defaults to
hljs-
.
- Type:
cancellationToken
- Type:
CancellationToken
- Description: The cancellation token for the asynchronous operation.
- Type:
Highlighted code.
ArgumentNullException
- Thrown if
code
is null.
- Thrown if
ArgumentException
- Thrown if
languageAlias
is not a valid HighlightJS language alias.
- Thrown if
InvocationException
- Thrown if a NodeJS error occurs.
ObjectDisposedException
- Thrown if this instance has been disposed or if an attempt is made to use one of its dependencies that has been disposed.
OperationCanceledException
- Thrown if
cancellationToken
is cancelled.
- Thrown if
string code = @"public string ExampleFunction(string arg)
{
// Example comment
return arg + ""dummyString"";
}";
string highlightedCode = await highlightJSService.HighlightAsync(code, "csharp");
Task<bool> IsValidLanguageAliasAsync(string languageAlias, CancellationToken cancellationToken = default)
Determines whether a language alias is valid.
languageAlias
- Type:
string
- Description: Language alias to validate. Visit https://github.com/highlightjs/highlight.js/tree/master/src/languages for the list of valid language aliases.
- Type:
true
if languageAlias
is a valid HighlightJS language alias. Otherwise, false
.
InvocationException
- Thrown if a NodeJS error occurs.
ObjectDisposedException
- Thrown if this instance has been disposed or if an attempt is made to use one of its dependencies that has been disposed.
OperationCanceledException
- Thrown if
cancellationToken
is cancelled.
- Thrown if
bool isValid = await highlightJSService.IsValidLanguageAliasAsync("csharp");
If you've used the javascript HighlightJS library before, you might have noticed that some of its features have been omitted in this wrapper. The following are the reasons for their omittance:
If ignore_illegals is false, the javascript HighlightJS library throws an error when invalid syntax is detected. This feature can be inaccurate because language definitions aren't always up to date.
The javascript HighlightJS library has an automatic language detection feature. It works by highlighting code using every language definition, then ranking languages based on the number of matches for each language definition (a language definition is essentially a set of regex expressions). This feature can be inaccurate, especially for short snippets.
The javascript HighlightJS library has a continuation feature. Essentially, it returns the context of every highlight call, allowing subsequent calls to continue highlighting based on the context of an earlier call. Making multiple highlight calls with the continuation feature is equivalent to concatenating the code and making a single highlight call. For this wrapper, a single highlight call is far more performant since it minimizes time spent on object marshalling and inter-process communication.
You can build and test this project in Visual Studio 2017/2019.
Jering.Web.SyntaxHighlighters.Prism - Use the Syntax Highlighter, Prism, from C#.
Jering.Markdig.Extensions.FlexiBlocks - A Collection of Flexible Markdig Extensions.
Jering.Javascript.NodeJS - Invoke Javascript in NodeJS, from C#.
Syntax highlighters add markup to code to facilitate styling. For example, the following code:
public string ExampleFunction(string arg)
{
// Example comment
return arg + "dummyString";
}
is transformed into the following markup by the syntax highlighter HighlightJS:
<span class="hljs-function"><span class="hljs-built_in">public</span> <span class="hljs-built_in">string</span> <span class="hljs-title">ExampleFunction</span>(<span class="hljs-params"><span class="hljs-built_in">string</span> arg</span>)
</span>{
<span class="hljs-comment">// Example comment</span>
<span class="hljs-built_in">return</span> arg + <span class="hljs-string">"dummyString"</span>;
}
HighlightJS is a a javascript library, which is ideal since syntax highlighting is often done client-side. There are however, situations where syntax highlighting can't or shouldn't be done client-side, for example:
- When generating AMP pages, since AMP pages cannot run scripts.
- When page load time is critical.
- When page size is critical.
This library allows syntax highlighting to be done by .Net server-side applications and tools like static site generators.
Contributions are welcome!
Follow @JeringTech for updates and more.