Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/LostBeard/SpawnDev.BlazorJS

Full Blazor WebAssembly and Javascript interop. Supports all Javascript data types and web browser APIs.
https://github.com/LostBeard/SpawnDev.BlazorJS

blazor blazor-webassembly browser-api csharp dom dotnet indexeddb interop javascript webassembly webbrowser webgl

Last synced: 2 months ago
JSON representation

Full Blazor WebAssembly and Javascript interop. Supports all Javascript data types and web browser APIs.

Awesome Lists containing this project

README

        

# SpawnDev.BlazorJS
[![NuGet](https://img.shields.io/nuget/dt/SpawnDev.BlazorJS.svg?label=SpawnDev.BlazorJS)](https://www.nuget.org/packages/SpawnDev.BlazorJS)

Full Blazor WebAssembly and Javascript interop. Create Javascript objects, access properties, call methods, and add/remove event handlers of any Javascript objects the .Net way without writing Javascript.

**[SpawnDev.BlazorJS.WebWorkers](https://github.com/LostBeard/SpawnDev.BlazorJS.WebWorkers)** is now in a separate repo [here.](https://github.com/LostBeard/SpawnDev.BlazorJS.WebWorkers)

[Live Demo](https://blazorjs.spawndev.com/)

### Supported .Net Versions
- Blazor WebAssembly .Net 6, 7, 8, and 9
- Tested VS Template: Blazor WebAssembly Standalone App
- Blazor United .Net 8 (in WebAssembly project only)
- Tested VS Template: Blazor Web App (Interactive WebAssembly mode without prerendering)

### Features:
- Supports all web browser [Web APIs](https://developer.mozilla.org/en-US/docs/Web/API)
- If we missed anything, open an issue and it will be updated ASAP.
- Supports all web browser Javascript data types
- Over 350 strongly typed [JSObject](#jsobjects) wrappers ([listed here](https://blazorjs.spawndev.com/JSObjectTypeInfo)) included in BlazorJS including DOM, Crypto, WebGL, WebRTC, Atomics, TypedArrays, and Promises allow direct interaction with Javascript
- Use Javascript libraries in Blazor without writing any Javascript code
- BlazorJSRuntime wraps the default JSRuntime adding additional functionality
- Create new Javascript objects directly from Blazor
- Get and set Javascript object properties as well as access methods
- Easily pass .Net methods to Javascript using ActionEvent, Callback.Create or Callback.CreateOne methods
- Easily wrap your Javascript objects for direct manipulation from Blazor (No javascript required!)
- Create a class that inherits from [JSObject](#jsobject-base-class) and define the methods, properties, events, and constructors.
- Supports [Promise](#promise), [Union](#union) method parameters, and passing [undefined](#undefined) to Javascript
- Supports Tuple, ValueTuple serialization to and from a Javascript Array
- Supports [null-conditional member access operator ?.](#null-conditional) in JS interop

# Issues and Feature requests
I'm here to help. If you find a bug or missing properties, methods, or Javascript objects please submit an issue [here](https://github.com/LostBeard/SpawnDev.BlazorJS/issues) on GitHub. I will help as soon as possible.

# BlazorJSRuntime
Getting started. Using BlazorJS requires 2 changes to your Program.cs.
- Add the BlazorJSRuntime service with builder.Services.AddBlazorJSRuntime()
- Initialize BlazorJSRuntime by calling builder.Build().BlazorJSRunAsync() instead of builder.Build().RunAsync()
- Supports [null-conditional member access operator ?.](#null-conditional) in JS interop

```cs
// ... other usings
using SpawnDev.BlazorJS;

var builder = WebAssemblyHostBuilder.CreateDefault(args);
builder.RootComponents.Add("#app");
builder.RootComponents.Add("head::after");

// Add SpawnDev.BlazorJS.BlazorJSRuntime
builder.Services.AddBlazorJSRuntime();

// build and Init using BlazorJSRunAsync (instead of RunAsync)
await builder.Build().BlazorJSRunAsync();
```

Inject into components
```cs
[Inject]
BlazorJSRuntime JS { get; set; }
```

Examples uses
```cs
// Get and Set
var innerHeight = JS.Get("window.innerHeight");
JS.Set("document.title", "Hello World!");

// Call
var item = JS.Call("localStorage.getItem", "itemName");
JS.CallVoid("addEventListener", "resize", Callback.Create(() => Console.WriteLine("WindowResized"), _callBacks));

// Attach events
using var window = JS.Get("window");
window.OnOffline += Window_OnOffline;

// AddEventListener and RemoveEventListener are supported on all EventTarget objects
window.AddEventListener("resize", Window_OnResize, true);

window.RemoveEventListener("resize", Window_OnResize, true);
```

## IMPORTANT NOTE - Async vs Sync Javascript calls
SpawnDev's BlazorJSRuntime behaves differently than Microsoft's Blazor JSRuntime. SpawnDev's BlazorJSRuntime is more of a 1 to 1 mapping to Javascript.

When calling Javascript methods that are not asynchronous and do not return a Promise you need to use the synchronous BlazorJSRuntime methods Call, CallVoid, or Get.
Unlike the default Blazor JSRuntime which would allow the use of InvokeAsync, you must use the synchronous BlazorJSRuntime methods.

Use synchronous BlazorJSRuntime calls for synchronous Javascript methods.
BlazorJSRuntime CallAsync would throw an error if used on the below Javascript method.

```js
// Javascript
function AddNum(num1, num2){
return num1 + num2;
}
```

```cs
// C#
var total = JS.Call("AddNum", 20, 22);
// total == 42 here
```

Use synchronous BlazorJSRuntime calls for asynchronous Javascript methods.
```js
// Javascript
async function AddNum(num1, num2){
return num1 + num2;
}
```

```cs
// C#
var total = await JS.Call>("AddNum", 20, 22);
// total == 42 here
```

```cs
// C#
var totalPromise = JS.Call>("AddNum", 20, 22);
var total = await totalPromise.ThenAsync();
// total == 42 here
```

Use asynchronous BlazorJSRuntime calls for asynchronous Javascript methods.
```js
// Javascript
async function AddNum(num1, num2){
return num1 + num2;
}
```

```cs
// C#
var total = await JS.CallAsync("AddNum", 20, 22);
// total == 42 here
```

Use asynchronous BlazorJSRuntime calls for methods that return a Promise.
```js
// Javascript
function AddNum(num1, num2){
return new Promise((resolve, reject)=>{
resolve(num1 + num2);
});
}
```

```cs
// C#
var total = await JS.CallAsync("AddNum", 20, 22);
// total == 42 here
```

## NULL Conditional
The BlazorJSRuntime now supports [null-conditional member access operator ?.](https://learn.microsoft.com/en-us/dotnet/csharp/language-reference/operators/member-access-operators#null-conditional-operators--and-).

Note: The null-conditional member access operator ?. is also known as the Elvis operator.

Example
```js
// Javascript
var fruit = {
name: 'apple',
color: 'red'
};
```

The below `JS.Get` would throw an error because `fruit.options` does not exist, and therefore we cannot access a property of it.
```cs
// C#
var size = JS.Get("fruit.options.size");
// never gets here due to error because `fruit.options` does not exist
```

Using a null conditional (the `?` in `fruit.options?.size`) prevents the error by allowing `fruit.options` to not exist (null or undefined.)
```cs
// C#
var size = JS.Get("fruit.options?.size");
// size == null here (default value for int?)
```

# IJSInProcessObjectReference extended

```cs
// Get Set
var window = JS.Get("window");
window.Set("myVar", 5);
var myVar = window.Get("myVar");

// Call
window.CallVoid("addEventListener", "resize", Callback.Create(() => Console.WriteLine("WindowResized")));
```

Create a new Javascript object
```cs
IJSInProcessObjectReference worker = JS.New("Worker", myWorkerScript);
```

# ActionEvent

Used throughout the JSObject collection, ActionEvent allows a clean .Net style way to add and remove .Net callbacks for Javascript events.

With ActionEvent the operands += and -= can be used to attach and detach .Net callbacks to Javascript events. All reference handling is done automatically when events are added and removed.

Example taken from the Window JSObject class which inherits from EventTarget.
```cs
// This is how ActionEvent is implemented in the Window class
public ActionEvent OnStorage { get => new ActionEvent("storage", AddEventListener, RemoveEventListener); set { } }
```

Example event attach detach
```cs
void AttachEventHandlersExample()
{
using var window = JS.Get("window");
// If this is the first time Window_OnStorage has been attached to an event a .Net reference is automatically created and held for future use and removal
window.OnStorage += Window_OnStorage;
// the window JSObject reference can safely be disposed as the .Net reference is attached to Window_OnStorage internally
}
void DetachEventHandlersExample()
{
using var window = JS.Get("window");
// If this is the last reference of Window_OnStorage being removed then the .Net reference will automatically be disposed.
// IMPORTANT - detaching is important for preventing resource leaks. .Net references are only released when the reference count reaches zero (same number of -= as += used)
window.OnStorage -= Window_OnStorage;
}
void Window_OnStorage(StorageEvent storageEvent)
{
Console.WriteLine($"StorageEvent");
}
```

### ActionEvent arguments are optional
Methods attached using ActionEvents are strongly typed and, like Javascript, all arguments are optional. This can improve performance as unused variables will not be brought into Blazor during the event.

Example event attach detach (from above) without using any callback arguments.
```cs
void AttachEventHandlersExample()
{
using var window = JS.Get("window");
window.OnStorage += Window_OnStorage;
}
void DetachEventHandlersExample()
{
using var window = JS.Get("window");
window.OnStorage -= Window_OnStorage;
}
// The method below is not using the optional StorageEvent argument
void Window_OnStorage()
{
Console.WriteLine($"StorageEvent");
}
```
### ActionEvent.On() and ActionEvent.Off()
ActionEvent has additional methods for attaching event handlers; `On` and `Off`. These methods provide support attaching event handlers with alternative parameter types.

Or use the `On` and `Off` methods:
```cs
void AttachEventHandlersExample()
{
using var window = JS.Get("window");
window.OnStorage.On(Window_OnStorage);
}
void DetachEventHandlersExample()
{
using var window = JS.Get("window");
window.OnStorage.Off(Window_OnStorage);
}
// The method below is not using the optional StorageEvent argument
void Window_OnStorage()
{
Console.WriteLine($"StorageEvent");
}
```

### Event handler parameters
Parameters of event handlers may be omitted if not required.

# FuncEvent
FuncEvent works just like ActionEvent but, as the name indicates, works with `Func` methods instead of `Action` methods to allow returning a value.

# Action and Func serialization
BlazorJS supports serialization of both Func and Action types. Internally the BlazorJS.Callback object is used. Serialized and deserialized Action and Func objects must call their DisposeJS() extension method to dispose the auto created and associated Callback and/or Function objects.

Action test from BlazorJSUnitTests.cs
```cs
var tcs = new TaskCompletionSource();
var callback = () =>
{
tcs.TrySetResult(true);
};
JS.CallVoid("setTimeout", callback, 100);
await tcs.Task;
callback.DisposeJS();
```

Func<,> test from BlazorJSUnitTests.cs
```cs
int testValue = 42;
var origFunc = new Func((val) =>
{
return val;
});
// set a global Javascript var to our Func
// if this is the first time this Func is passed to Javascript a Callback will be created and associated to this Func for use in future serialization
// the auto created Callback must be disposed by calling the extension method Func.DisposeJS()
JS.Set("_funcCallback", origFunc);
// read back in our Func as an Func
// internally a Javascript Function reference is created and associated with this Func.
// the auto created Function must be disposed by calling the extension method Func.DisposeJS()
var readFunc = JS.Get>("_funcCallback");
var readVal = readFunc(testValue);
if (readVal != testValue) throw new Exception("Unexpected result");
// dispose the Function created and associated with the read Func
readFunc.DisposeJS();
// dispose the Callback created and associated with the original Func
origFunc.DisposeJS();
```

# Callback

The Callback object is used to support Action and Func serialization. It can be used for a bit more control over the lifetime of you callbacks. Pass methods to Javascript using the Callback.Create and Callback.CreateOne methods. These methods use type arguments to set the types expected for incoming arguments (if any) and the expected return type (if any.) async methods are passed as Promises.

Pass lambda callbacks to Javascript
```cs
JS.Set("testCallback", Callback.Create((string strArg) => {
Console.WriteLine($"Javascript sent: {strArg}");
// this prints "Hello callback!"
}));
```
```js
// in Javascript
testCallback('Hello callback!');
```

Pass method callbacks to Javascript
```cs
string SomeNetFn(string input){
return $"Recvd: {input}";
}

JS.Set("someNetFn", Callback.CreateOne(SomeNetFn));
```
```js
// in Javascript
someNetFn('Hello callback!');

// prints
Recvd: Hello callback!
```

Pass async method callbacks to Javascript
Under the hood, BlazorJS is returning a Promise to Javascript when the method is called

```cs
async Task SomeNetFnAsync(string input)
{
await Task.Delay(1000);
return $"Recvd: {input}";
}

JS.Set("someNetFnAsync", Callback.CreateOne(SomeNetFnAsync));
```
```js
// in Javascript
await someNetFnAsync('Hello callback!');

// prints
Recvd: Hello callback!
```

# JSObjects
Over 350 Javascript types are ready to go in **SpawnDev.BlazorJS**. The classes are designed to match the Javascript [Web API interfaces](https://developer.mozilla.org/en-US/docs/Web/API#interfaces) as closely as possible. Below are some examples.

## HTMLVideoElement
From [HTMLVideoElement on MDN](https://developer.mozilla.org/en-US/docs/Web/API/HTMLVideoElement):
> Implemented by the \ element, the HTMLVideoElement interface provides special properties and methods for manipulating video objects. It also inherits properties and methods of HTMLMediaElement and HTMLElement.

Example using SpawnDev.BlazorJS.JSObjects.HTMLVideoElement

From HTMLVideoElementExample.cs
```cs
@page "/HTMLVideoElementExample"
@implements IDisposable





Source: @videoName


Duration: @duration.ToString()


Metadata: @metadata


@foreach (var video in videos)
{
@video.Key
}


@((MarkupString)log)

@code {
[Inject]
BlazorJSRuntime JS { get; set; }
ElementReference? videoElRef;
HTMLVideoElement? videoEl = null;
TimeSpan duration = TimeSpan.Zero;
string videoName = "";
string metadata = "";
string log = "";
Dictionary videos = new Dictionary
{
{ "Elephants Dream", "http://commondatastorage.googleapis.com/gtv-videos-bucket/sample/ElephantsDream.mp4" },
{ "Big Buck Bunny", "http://commondatastorage.googleapis.com/gtv-videos-bucket/sample/BigBuckBunny.mp4" },
{ "Tears Of Steel", "http://commondatastorage.googleapis.com/gtv-videos-bucket/sample/TearsOfSteel.mp4" },
{ "Sintel", "http://commondatastorage.googleapis.com/gtv-videos-bucket/sample/Sintel.mp4" },
{ "None", "" },
};
protected override void OnAfterRender(bool firstRender)
{
if (firstRender)
{
videoEl = (HTMLVideoElement)videoElRef!;
videoEl.OnLoadedMetadata += VideoEl_OnLoadedMetadata;
videoEl.OnAbort += VideoEl_OnAbort;
videoEl.OnError += VideoEl_OnError;
}
}
void SetSource(string name, string source)
{
if (videoEl == null) return;
Log($"SetSource: {name}");
videoName = name;
videoEl.Src = source;
StateHasChanged();
}
void VideoEl_OnLoadedMetadata()
{
Log("VideoEl_OnLoadedMetadata");
metadata = $"{videoEl!.VideoWidth}x{videoEl!.VideoHeight}";
duration = TimeSpan.FromSeconds(videoEl!.Duration ?? 0);
StateHasChanged();
}
void VideoEl_OnError()
{
Log("VideoEl_OnError");
}
void VideoEl_OnAbort()
{
Log("VideoEl_OnAbort");
metadata = $"{videoEl!.VideoWidth}x{videoEl!.VideoHeight}";
duration = TimeSpan.FromSeconds(videoEl!.Duration ?? 0);
StateHasChanged();
}
public void Dispose()
{
if (videoEl != null)
{
videoEl.OnLoadedMetadata -= VideoEl_OnLoadedMetadata;
videoEl.OnAbort -= VideoEl_OnAbort;
videoEl.OnError -= VideoEl_OnError;
videoEl.Dispose();
videoEl = null;
}
}
void Log(string message)
{
log += $"{message}
";
}
}
```

## HTMLCanvasElement
***Example coming soon***

## Storage - LocalStorage, SessionStorage
```cs
[Inject]
BlazorJSRuntime JS { get; set; }

override void OnInitialized()
{
using Storage localStorage = JS.Get("localStorage");
localStorage.SetItem("myKey", "myValue");
var myValue = localStorage.GetItem("myKey");
// myValue == "myValue"
}
```

## IndexedDB
From [IndexedDB on MDN](https://developer.mozilla.org/en-US/docs/Web/API/IndexedDB_API):
> IndexedDB is a low-level API for client-side storage of significant amounts of structured data, including files/blobs.

The below code was written to test various features of the IndexedDB API and this code specifically tests using a Tuple as an ObjectStore key.

```cs
[Inject]
BlazorJSRuntime JS { get; set; }

public class Fruit
{
public (byte[], long) MyKey { get; set; }
public string Name { get; set; }
public string Color { get; set; }
}

override void OnInitialized()
{
var dbName = "garden_tuple";
var dbStoreName = "fruit";
// Get the global IDBFactory (equivalent to 'JS.Get("indexedDB")')
using var idbFactory = new IDBFactory();
var idb = await idbFactory.OpenAsync(dbName, 2, (evt) =>
{
// upgrade needed
using var request = evt.Target;
using var db = request.Result;
var stores = db.ObjectStoreNames;
if (!stores.Contains(dbStoreName))
{
using var store = db.CreateObjectStore(dbStoreName, new IDBObjectStoreCreateOptions { KeyPath = "name" });
store.CreateIndex<(byte[], long)>("tuple_index", "myKey");
}
});

// transaction
using var tx = idb.Transaction(dbStoreName, "readwrite");
using var objectStore = tx.ObjectStore(dbStoreName);

// add some data
await objectStore.PutAsync(new Fruit { Name = "apple", Color = "red", MyKey = (new byte[] { 1, 2, 3 }, 5) });
await objectStore.PutAsync(new Fruit { Name = "orange", Color = "orange", MyKey = (new byte[] { 1, 2, 5 }, 5) });
await objectStore.PutAsync(new Fruit { Name = "lemon", Color = "yellow", MyKey = (new byte[] { 1, 2, 5 }, 5) });
await objectStore.PutAsync(new Fruit { Name = "lime", Color = "green", MyKey = (new byte[] { 33, 33, 45 }, 5) });

// get an IDBIndex
using var myIndex = objectStore.Index<(byte[], long)>("tuple_index");

// create a range using ValueTuple type
using var range = IDBKeyRange<(byte[], long)>.Bound((new byte[] { 0, 0, 0 }, 0), (new byte[] { 5, 5, 5 }, long.MaxValue));

var included = range.Includes((new byte[] { 1, 2, 4 }, 5));

var cmpRet0 = idbFactory.Cmp<(byte[], long)>((new byte[] { 1, 2, 3 }, 6), (new byte[] { 1, 2, 3 }, 5));
var cmpRet1 = idbFactory.Cmp<(byte[], long)>((new byte[] { 1, 2, 3 }, 5), (new byte[] { 1, 2, 3 }, 5));
var cmpRet2 = idbFactory.Cmp<(byte[], long)>((new byte[] { 1, 2, 2 }, 5), (new byte[] { 1, 2, 3 }, 5));
var cmpRet3 = idbFactory.Cmp<(byte[], long)>((new byte[] { 1, 2, 4 }, 5), (new byte[] { 1, 2, 3 }, 4));

// getAll on IDBIndex using the above range
using var getAll = await myIndex.GetAllAsync(range);

// below prints "apple", "orange", "lemon"
// the "lime" entry's byte[] is outside of our range and therefore not included
foreach (var item in getAll.ToArray())
{
JS.Log(item.Name);
}

// get on IDBIndex. returns null if not found.
var get = await myIndex.GetAsync((new byte[] { 1, 2, 5 }, 5));
JS.Log("get", get);

// getAll on ObjectStore
var getAllStore = await objectStore.GetAllAsync();
JS.Log("getAllStore", getAllStore);

// IDBCursor iteration
using var cursor = await myIndex.OpenCursorAsync();
var hasData = cursor != null;
while (hasData)
{
var canCont1 = await cursor.CanContinue();
JS.Log("Entry", cursor!.Value);
hasData = await cursor!.ContinueAsync();
var canCont2 = await cursor.CanContinue();
var nmt = true;
}
JS.Log("Done");
}
```

## Cache
From [Cache on MDN](https://developer.mozilla.org/en-US/docs/Web/API/Cache)
> The Cache interface provides a persistent storage mechanism for Request / Response object pairs that are cached in long lived memory.

***Example coming soon***

## TypedArray
SpawnDev.BlazorJS supports all TypedArray types.
***Example coming soon***

## Atomics
From [Atomics on MDN](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Atomics):
> The Atomics namespace object contains static methods for carrying out atomic operations. They are used with SharedArrayBuffer and ArrayBuffer objects.

***Example coming soon***

# JSObject Base Class

JSObjects are wrappers around IJSInProcessReference objects that can be passed to and from Javascript and allow strongly typed access to the underlying object.

JSObject type wrapper example
```cs
// create a class for your Javascript object that inherits from JSObject
public class Window : JSObject
{
// required constructor
public Window(IJSInProcessObjectReference _ref) : base(_ref) { }
public string Name { get => JSRef.Get("name"); set => JSRef.Set("name", value); }
public void Alert(string msg = "") => JSRef.CallVoid(msg);
// ...
}

// use the JSObject class to interact with the Javascript object
public void JSObjectClassTest() {
var w = JS.Get("window");
var randName = Guid.NewGuid().ToString();
// directly set the window.name property
w.Name = randName;
// verify the read back
if (w.Name != randName) throw new Exception("Interface property set/get failed");
}
```

Use the extended functions of IJSInProcessObjectReference to work with Javascript objects or
use the growing library of over 350 of the most common Javascript objects, including ones for
Window, Document, Storage (localStorage and sessionStorage), WebGL, WebRTC, and more in
SpawnDev.BlazorJS.JSObjects. JSObjects are wrappers around IJSInProcessObjectReference that
allow strongly typed use.

Below shows a section of the SpawnDev.BlazorJS.JSObjects.Window class. Window's base type, EventTarget, inherits from JSObject.
```cs
public class Window : EventTarget {
// all JSObject types must have this constructor
public Window(IJSInProcessObjectReference _ref) : base(_ref) { }
// here is a property with both getter and setter
public string? Name { get => JSRef.Get("name"); set => JSRef.Set("name", value); }
// here is a read only property that returns another JSObject type
public Storage LocalStorage => JSRef.Get("localStorage");
// here are methods
public long SetTimeout(Callback callback, double delay) => JSRef.Call("setTimeout", callback, delay);
public void ClearTimeout(long requestId) => JSRef.CallVoid("clearTimeout", requestId);
// ...
}
```

Below the JSObject derived Window class is used
```cs
// below the JSObject derived Window class is used
using var window = JS.Get("window");
var randName = Guid.NewGuid().ToString();
// set and get properties
window.Name = randName;
var name = window.Name;
// call methods
window.Alert("Hello!");
```

## Promise
SpawnDev.BlazorJS.JSObjects.Promise - is a JSObject wrapper for the Javascript Promise class.
Promises can be created in .Net to wrap async methods or Tasks. They are essentially Javascript's version of Task.

Create Promise from lambda method
```cs
var promise = new Promise(async () => {
await Task.Delay(5000);
});
// pass to Javascript api

```
Create Promise from lambda method with return value
```cs
var promise = new Promise(async () => {
await Task.Delay(5000);
return "Hello world!";
});
// pass to Javascript api
```
Create Promise from Task
```cs
var taskSource = new TaskCompletionSource();
var promise = new Promise(taskSource.Task);
// pass to Javascript api

// then later resolve
taskSource.TrySetResult("Hello world!");
```

Below is a an example that uses Promises to utilize the [Web Locks API](https://developer.mozilla.org/en-US/docs/Web/API/Web_Locks_API) (**Note**: The below code is designed to demonstrate the use of a Promise. This is not the recommended way of using LockManager. See [WebWorkerService.Locks](#webworkerservicelocks) for more info.)

```cs
using var navigator = JS.Get("navigator");
using var locks = navigator.Locks;

Console.WriteLine($"lock: 1");

using var waitLock = locks.Request("my_lock", Callback.CreateOne((Lock lockObj) => new Promise(async () => {
Console.WriteLine($"lock acquired 3");
await Task.Delay(5000);
Console.WriteLine($"lock released 4");
})));

using var waitLock2 = locks.Request("my_lock", Callback.CreateOne((Lock lockObj) => new Promise(async () => {
Console.WriteLine($"lock acquired 5");
await Task.Delay(5000);
Console.WriteLine($"lock released 6");
})));

Console.WriteLine($"lock: 2");
```

## Custom JSObjects
Implement your own JSObject classes for Javascript objects not already available in the BlazorJS.JSObjects library.

Instead of this (simple but not as reusable)
```cs
var audio = JS.New("Audio", "https://some_audio_online");
audio.CallVoid("play");
```

You can do this...
Create a custom Audio JSObject wrapper (Example only. Already exists.)
```cs
public class Audio : JSObject
{
// deserialization constructor
public Audio(IJSInProcessObjectReference _ref) : base(_ref) { }

// constructor that accepts a string url
public Audio(string url) : base(JS.New("Audio", url)) { }

// method decalaration
public void Play() => JSRef.CallVoid("play");
}
```

Then use the Audio JSObject
```cs
var audio = new Audio("https://some_audio_online");
audio.Play();
```

# Union
## Use the Union\ type with method parameters for strong typing while allowing unrelated types just like in TypeScript.

```cs
void UnionTypeTestMethod(string varName, Union? unionTypeValue)
{
JS.Set(varName, unionTypeValue);
}

var stringValue = "Hello world!";
UnionTypeTestMethod("_stringUnionValue", stringValue);
if (stringValue != JS.Get("_stringUnionValue")) throw new Exception("Unexpected result");

var boolValue = true;
UnionTypeTestMethod("_boolUnionValue", boolValue);
if (boolValue != JS.Get("_boolUnionValue")) throw new Exception("Unexpected result");
```

# Undefinable
## Use Undefinable\ type to pass undefined to Javascript
Some Javascript API calls may have optional parameters that behave differently depending on if you pass a null versus undefined. You can now retain strong typing on JSObject method calls and support passing undefined for JSObject parameters.

Undefinable\ type.

Example from Test app unit tests
```cs
// an example method with a parameter that can also be null or undefined
// T of Undefinable must be nullable
void MethodWithUndefinableParams(string varName, Undefinable? window)
{
JS.Set(varName, window);
}

bool? w = false;
// test to show the value is passed normally
MethodWithUndefinableParams("_willBeDefined2", w);
bool? r = JS.Get("_willBeDefined2");
if (r != w) throw new Exception("Unexpected result");

w = null;
// null defaults to passing as undefined
MethodWithUndefinableParams("_willBeUndefined2", w);
if (!JS.IsUndefined("_willBeUndefined2")) throw new Exception("Unexpected result");

// if you need to pass null to an Undefinable parameter use Undefinable.Null
MethodWithUndefinableParams("_willBeNull2", Undefinable.Null);
if (JS.IsUndefined("_willBeNull2")) throw new Exception("Unexpected result");

// another way to pass undefined
MethodWithUndefinableParams("_willAlsoBeUndefined2", Undefinable.Undefined);
if (!JS.IsUndefined("_willAlsoBeUndefined2")) throw new Exception("Unexpected result");
```

If using JSObjects you can also use JSObject.Undefined\ to create an instance that will be passed to Javascript as undefined.

```cs
// Create an instance of the Window JSObject class that is revived in Javascript as undefined
var undefinedWindow = JSObject.Undefined();
// undefinedWindow is an instance of Window that is revived in Javascript as undefined
JS.Set("_undefinedWindow", undefinedWindow);
var isUndefined = JS.IsUndefined("_undefinedWindow");
// isUndefined == true here
```

# SpawnDev.BlazorJS.WebWorkers
- SpawnDev.BlazorJS.WebWorkers has moved to its own repo: [SpawnDev.BlazorJS.WebWorkers](https://github.com/LostBeard/SpawnDev.BlazorJS.WebWorkers)

# Blazor Web App compatibility
.Net 8 introduced a new hosting model that allows mixing [Blazor server render mode](https://learn.microsoft.com/en-us/aspnet/core/blazor/components/render-modes?view=aspnetcore-8.0#interactive-server-side-rendering-interactive-ssr) and [Blazor WebAssembly render mode](https://learn.microsoft.com/en-us/aspnet/core/blazor/components/render-modes?view=aspnetcore-8.0#client-side-rendering-csr). [Prerendering](https://learn.microsoft.com/en-us/aspnet/core/blazor/components/render-modes?view=aspnetcore-8.0#prerendering) was also added to improve initial rendering times. "Prerendering is the process of initially rendering page content on the server without enabling event handlers for rendered controls."

One of the primary goals of SpawnDev.BlazorJS is to give [Web API](https://developer.mozilla.org/en-US/docs/Web/API) access to Blazor WebAssembly that mirrors Javascript's own Web API. This includes calling conventions. For example, a call that is synchronous in Javascript is synchronous in Blazor, an asynchronous call is asynchronous. To provide that, SpawnDev.BlazorJS requires access to Microsoft's IJSInProcessRuntime and IJSInProcessRuntime is only available in Blazor WebAssembly.

## Compatible ```Blazor Web App``` options:
As of version 2.5.11 the BlazorJSRuntime service can be registered on the server the same way it is on the client. Support for this was enabled to allow prerendering if needed. While it can be registered on the server, the BlazorJSRuntime is not functional unless running in WebAssembly. A component that uses `BlazorJSRuntime` can use the service property `IsBrowser` or `OperatingSystem.IsBrowser()` to determine if the code is running in a browser. To give components a functional BlazorJSRuntime, let Blazor know that those components must be rendered with WebAssembly. How this is done depends on your project settings.

### ```Interactive render mode``` - ```Auto (Server and WebAssembly)``` or ```WebAssembly```

### ```Interactivity location``` - ```Per page/component```

In the Server project ```App.razor```:
```html

```

In WebAssembly pages and components that require SpawnDev.BlazorJS (prerender optional):
```cs
@rendermode @(new InteractiveWebAssemblyRenderMode(prerender: false))
```

### ```Interactivity location``` - ```Global```
In the Server project ```App.razor``` (prerender optional):
```html

```

# IDisposable
NOTE: The above code shows quick examples. Some objects implement IDisposable, such as JSObject, Callback, and IJSInProcessObjectReference types.

JSObject types will dispose of their IJSInProcessObjectReference object when their finalizer is called if not previously disposed.

Callback types must be disposed unless created with the Callback.CreateOne method, in which case they will dispose themselves after the first callback. Disposing a Callback prevents it from being called.

IJSInProcessObjectReference does not dispose of interop resources with a finalizer and MUST be disposed when no longer needed. Failing to dispose these will cause memory leaks.

IDisposable objects returned from a WebWorker or SharedWorker service are automatically disposed after the data has been sent to the calling thread.

# Support for You
Issues can be reported [here](https://github.com/LostBeard/SpawnDev.BlazorJS/issues) on GitHub. Create a new [discussion](https://github.com/LostBeard/SpawnDev.BlazorJS/discussions) to show off your projects and post your ideas. We are always here to help.

# Support for Us
Sponsor us via Github Sponsors to give us more time to work on SpawnDev.BlazorJS.WebWorkers and other open source projects. Or buy us a cup of coffee via Paypal. All support is greatly appreciated! ♥

[![GitHub Sponsor](https://img.shields.io/github/sponsors/LostBeard?label=Sponsor&logo=GitHub&color=%23fe8e86)](https://github.com/sponsors/LostBeard)
[![Donate](https://img.shields.io/badge/Donate-PayPal-green.svg)](https://www.paypal.com/cgi-bin/webscr?cmd=_s-xclick&hosted_button_id=7QTATH4UGGY9U)

# Thanks
Thank you to everyone who has helped support SpawnDev.BlazorJS and related projects financially, by filing issues, and by improving the code. Every little contribution helps!

# Demos
BlazorJS and WebWorkers Demo
https://blazorjs.spawndev.com/

Current site under development using Blazor WASM
https://www.spawndev.com/