@nativescript/windows 0.1.0-alpha.1 → 0.1.0-alpha.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build.ps1 +34 -1
- package/framework/__PROJECT_NAME__/__PROJECT_NAME__.csproj +39 -6
- package/framework/dotnet-bridge/Bridge.cs +353 -0
- package/framework/dotnet-bridge/DotNetBridge.csproj +20 -0
- package/framework/libs/arm64/nativescript.dll +0 -0
- package/framework/libs/devtools/arm64/nativescript.dll +0 -0
- package/framework/libs/devtools/x64/nativescript.dll +0 -0
- package/framework/libs/x64/nativescript.dll +0 -0
- package/package.json +3 -2
package/build.ps1
CHANGED
|
@@ -27,7 +27,8 @@
|
|
|
27
27
|
param(
|
|
28
28
|
[switch]$SkipArm64,
|
|
29
29
|
[switch]$SkipRelease,
|
|
30
|
-
[switch]$SkipDevtools
|
|
30
|
+
[switch]$SkipDevtools,
|
|
31
|
+
[switch]$SkipDotnet
|
|
31
32
|
)
|
|
32
33
|
|
|
33
34
|
Set-StrictMode -Version Latest
|
|
@@ -69,6 +70,38 @@ function Build-Crate {
|
|
|
69
70
|
}
|
|
70
71
|
}
|
|
71
72
|
|
|
73
|
+
# ── DotNet Bridge ─────────────────────────────────────────────────────────────
|
|
74
|
+
# Copies the dotnet-bridge source (csproj + .cs files) into
|
|
75
|
+
# template/framework/dotnet-bridge/ so it ships inside the npm package.
|
|
76
|
+
# The actual `dotnet publish` is deferred to the app's MSBuild process —
|
|
77
|
+
# see the PublishDotNetBridge target in __PROJECT_NAME__.csproj.
|
|
78
|
+
|
|
79
|
+
if (-not $SkipDotnet) {
|
|
80
|
+
Write-Host "`n=== DotNet Bridge (source copy) ===" -ForegroundColor Cyan
|
|
81
|
+
$BridgeSrc = Join-Path $RepoRoot "dotnet-bridge"
|
|
82
|
+
$BridgeDest = Join-Path $ScriptDir "framework\dotnet-bridge"
|
|
83
|
+
|
|
84
|
+
# Wipe the destination so stale files don't linger between runs.
|
|
85
|
+
if (Test-Path $BridgeDest) { Remove-Item -Recurse -Force $BridgeDest }
|
|
86
|
+
$null = New-Item -ItemType Directory -Force -Path $BridgeDest
|
|
87
|
+
|
|
88
|
+
# Copy only source files; exclude build/publish artefacts.
|
|
89
|
+
Get-ChildItem -Path $BridgeSrc -Recurse |
|
|
90
|
+
Where-Object {
|
|
91
|
+
$rel = $_.FullName.Substring($BridgeSrc.Length + 1)
|
|
92
|
+
$rel -notmatch '^(bin|obj|publish)(\\|$)'
|
|
93
|
+
} |
|
|
94
|
+
ForEach-Object {
|
|
95
|
+
$dest = Join-Path $BridgeDest $_.FullName.Substring($BridgeSrc.Length + 1)
|
|
96
|
+
if ($_.PSIsContainer) {
|
|
97
|
+
$null = New-Item -ItemType Directory -Force -Path $dest
|
|
98
|
+
} else {
|
|
99
|
+
Copy-Item -Force $_.FullName $dest
|
|
100
|
+
Write-Host " copied $($_.FullName.Substring($BridgeSrc.Length + 1))"
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
|
|
72
105
|
# ── Targets ───────────────────────────────────────────────────────────────────
|
|
73
106
|
|
|
74
107
|
$Targets = @(
|
|
@@ -19,6 +19,14 @@
|
|
|
19
19
|
<NSWindowsRoot>$(MSBuildProjectDirectory)\..</NSWindowsRoot>
|
|
20
20
|
</PropertyGroup>
|
|
21
21
|
|
|
22
|
+
<PropertyGroup>
|
|
23
|
+
<!-- Derived library platform for selecting prebuilt native libs.
|
|
24
|
+
Prefer $(Platform); fall back to trimming 'win-' from $(RuntimeIdentifier); default to x64. -->
|
|
25
|
+
<NSLibPlatform Condition="'$(Platform)' != ''">$(Platform)</NSLibPlatform>
|
|
26
|
+
<NSLibPlatform Condition="'$(NSLibPlatform)' == '' and '$(RuntimeIdentifier)' != ''">$([System.Text.RegularExpressions.Regex]::Replace('$(RuntimeIdentifier)','^win-',''))</NSLibPlatform>
|
|
27
|
+
<NSLibPlatform Condition="'$(NSLibPlatform)' == ''">x64</NSLibPlatform>
|
|
28
|
+
</PropertyGroup>
|
|
29
|
+
|
|
22
30
|
<ItemGroup>
|
|
23
31
|
<AppxManifest Include="Package.appxmanifest" />
|
|
24
32
|
<Content Include="Properties\Default.rd.xml" />
|
|
@@ -33,18 +41,43 @@
|
|
|
33
41
|
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
|
|
34
42
|
</Content>
|
|
35
43
|
|
|
36
|
-
<!-- Debug: devtools-enabled runtime DLL -->
|
|
37
|
-
<Content Include="$(NSWindowsRoot)\libs\devtools\$(
|
|
38
|
-
Condition="'$(Configuration)' == 'Debug'">
|
|
44
|
+
<!-- Debug: devtools-enabled runtime DLL (platform fallback via NSLibPlatform) -->
|
|
45
|
+
<Content Include="$(NSWindowsRoot)\libs\devtools\$(NSLibPlatform)\nativescript.dll"
|
|
46
|
+
Condition="'$(Configuration)' == 'Debug' and Exists('$(NSWindowsRoot)\libs\devtools\$(NSLibPlatform)\nativescript.dll')">
|
|
39
47
|
<Link>nativescript.dll</Link>
|
|
40
48
|
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
|
|
41
49
|
</Content>
|
|
42
50
|
|
|
43
|
-
<!-- Release: production runtime DLL -->
|
|
44
|
-
<Content Include="$(NSWindowsRoot)\libs\$(
|
|
45
|
-
Condition="'$(Configuration)' != 'Debug'">
|
|
51
|
+
<!-- Release: production runtime DLL (platform fallback via NSLibPlatform) -->
|
|
52
|
+
<Content Include="$(NSWindowsRoot)\libs\$(NSLibPlatform)\nativescript.dll"
|
|
53
|
+
Condition="'$(Configuration)' != 'Debug' and Exists('$(NSWindowsRoot)\libs\$(NSLibPlatform)\nativescript.dll')">
|
|
46
54
|
<Link>nativescript.dll</Link>
|
|
47
55
|
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
|
|
48
56
|
</Content>
|
|
57
|
+
|
|
58
|
+
<!-- .NET bridge — published on first build by the target below; enables ns.dotnet.*
|
|
59
|
+
APIs from JavaScript. Must land at dotnet-bridge\publish\ relative to the
|
|
60
|
+
executable so the Rust runtime can locate DotNetBridge.dll via
|
|
61
|
+
AppContext.BaseDirectory. Condition guards against a missing source tree. -->
|
|
62
|
+
<Content Include="$(NSWindowsRoot)\dotnet-bridge\publish\**\*"
|
|
63
|
+
Condition="Exists('$(NSWindowsRoot)\dotnet-bridge\publish')">
|
|
64
|
+
<Link>dotnet-bridge\publish\%(RecursiveDir)%(Filename)%(Extension)</Link>
|
|
65
|
+
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
|
|
66
|
+
</Content>
|
|
49
67
|
</ItemGroup>
|
|
68
|
+
|
|
69
|
+
<!--
|
|
70
|
+
Publishes DotNetBridge.dll before the app build so the Content glob above has
|
|
71
|
+
something to pick up. Inputs/Outputs make this incremental: MSBuild skips the
|
|
72
|
+
Exec entirely when DotNetBridge.dll is newer than both source files.
|
|
73
|
+
-->
|
|
74
|
+
<Target Name="PublishDotNetBridge"
|
|
75
|
+
BeforeTargets="Build"
|
|
76
|
+
Inputs="$(NSWindowsRoot)\dotnet-bridge\DotNetBridge.csproj;
|
|
77
|
+
$(NSWindowsRoot)\dotnet-bridge\Bridge.cs"
|
|
78
|
+
Outputs="$(NSWindowsRoot)\dotnet-bridge\publish\DotNetBridge.dll"
|
|
79
|
+
Condition="Exists('$(NSWindowsRoot)\dotnet-bridge\DotNetBridge.csproj')">
|
|
80
|
+
<Message Text="[NativeScript] Publishing DotNetBridge..." Importance="high" />
|
|
81
|
+
<Exec Command="dotnet publish "$(NSWindowsRoot)\dotnet-bridge\DotNetBridge.csproj" -c Release -o "$(NSWindowsRoot)\dotnet-bridge\publish" --no-self-contained /nologo" />
|
|
82
|
+
</Target>
|
|
50
83
|
</Project>
|
|
@@ -0,0 +1,353 @@
|
|
|
1
|
+
using System;
|
|
2
|
+
using System.Collections;
|
|
3
|
+
using System.Collections.Concurrent;
|
|
4
|
+
using System.Collections.Generic;
|
|
5
|
+
using System.Linq;
|
|
6
|
+
using System.Reflection;
|
|
7
|
+
using System.Runtime.InteropServices;
|
|
8
|
+
using System.Text.Json;
|
|
9
|
+
using System.Text.Json.Nodes;
|
|
10
|
+
using System.Threading;
|
|
11
|
+
using System.Threading.Tasks;
|
|
12
|
+
|
|
13
|
+
namespace NativeScriptBridge;
|
|
14
|
+
|
|
15
|
+
/// <summary>
|
|
16
|
+
/// Reflection-based .NET dispatch bridge — UTF-8 ABI edition.
|
|
17
|
+
///
|
|
18
|
+
/// Invoke() receives a UTF-8 byte slice (no string allocation on the Rust side),
|
|
19
|
+
/// dispatches via reflection, and serialises the response directly to a UTF-8
|
|
20
|
+
/// byte array via JsonSerializer.SerializeToUtf8Bytes(). This removes the
|
|
21
|
+
/// UTF-16 encode/decode round-trip of the original char* ABI.
|
|
22
|
+
///
|
|
23
|
+
/// Request JSON schema
|
|
24
|
+
/// -------------------
|
|
25
|
+
/// Static call: { "assembly": "System", "typeName": "System.Diagnostics.Stopwatch", "method": "StartNew", "args": [] }
|
|
26
|
+
/// Constructor: { "assembly": "System", "typeName": "System.Text.StringBuilder", "method": ".ctor", "args": [128] }
|
|
27
|
+
/// Instance call: { "handle": 3, "method": "Stop", "args": [] }
|
|
28
|
+
/// Property get: { "assembly": "...", "typeName": "...", "method": "get_Now", "args": [] }
|
|
29
|
+
/// Property set: { "handle": 3, "method": "set_IsEnabled", "args": [true] }
|
|
30
|
+
/// Release: { "handle": 3, "method": "__release", "args": [] }
|
|
31
|
+
///
|
|
32
|
+
/// Response JSON schema
|
|
33
|
+
/// --------------------
|
|
34
|
+
/// Primitive / string: { "result": 42.5 }
|
|
35
|
+
/// Managed object: { "result": { "__handle": 7, "__type": "System.Diagnostics.Stopwatch" } }
|
|
36
|
+
/// Array / enumerable: { "result": [1, 2, 3] }
|
|
37
|
+
/// Error: { "error": "Method not found" }
|
|
38
|
+
/// </summary>
|
|
39
|
+
public static class Bridge
|
|
40
|
+
{
|
|
41
|
+
private static readonly ConcurrentDictionary<int, object?> s_handles = new();
|
|
42
|
+
private static int s_nextHandle;
|
|
43
|
+
|
|
44
|
+
private static readonly ConcurrentDictionary<string, Type?> s_typeCache = new();
|
|
45
|
+
private static readonly ConcurrentDictionary<string, MethodInfo?> s_methodCache = new();
|
|
46
|
+
private static readonly ConcurrentDictionary<string, PropertyInfo?> s_propertyCache = new();
|
|
47
|
+
|
|
48
|
+
// ── exported entry points ────────────────────────────────────────────────
|
|
49
|
+
|
|
50
|
+
/// Called by the Rust runtime. Request arrives as a raw UTF-8 byte slice;
|
|
51
|
+
/// response is written as a UTF-8 byte buffer allocated with Marshal.AllocHGlobal.
|
|
52
|
+
[UnmanagedCallersOnly(EntryPoint = "Invoke",
|
|
53
|
+
CallConvs = [typeof(System.Runtime.CompilerServices.CallConvCdecl)])]
|
|
54
|
+
public static unsafe int Invoke(
|
|
55
|
+
byte* requestPtr, int requestLen,
|
|
56
|
+
byte** responsePtr, int* responseLenPtr)
|
|
57
|
+
{
|
|
58
|
+
byte[] responseBytes;
|
|
59
|
+
try
|
|
60
|
+
{
|
|
61
|
+
var requestSpan = new ReadOnlySpan<byte>(requestPtr, requestLen);
|
|
62
|
+
var req = JsonSerializer.Deserialize<InvokeRequest>(requestSpan, JsonOptions.Default)!;
|
|
63
|
+
var result = Dispatch(req);
|
|
64
|
+
responseBytes = JsonSerializer.SerializeToUtf8Bytes(result, JsonOptions.Default);
|
|
65
|
+
}
|
|
66
|
+
catch (Exception ex)
|
|
67
|
+
{
|
|
68
|
+
var errResult = new InvokeResult(null, Unwrap(ex).Message);
|
|
69
|
+
responseBytes = JsonSerializer.SerializeToUtf8Bytes(errResult, JsonOptions.Default);
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
WriteResponse(responseBytes, responsePtr, responseLenPtr);
|
|
73
|
+
return 0;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
/// Frees the response buffer allocated by Invoke.
|
|
77
|
+
[UnmanagedCallersOnly(EntryPoint = "Free",
|
|
78
|
+
CallConvs = [typeof(System.Runtime.CompilerServices.CallConvCdecl)])]
|
|
79
|
+
public static unsafe void Free(byte* ptr)
|
|
80
|
+
{
|
|
81
|
+
if (ptr != null)
|
|
82
|
+
Marshal.FreeHGlobal((IntPtr)ptr);
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
// ── dispatch ─────────────────────────────────────────────────────────────
|
|
86
|
+
|
|
87
|
+
private static InvokeResult Dispatch(InvokeRequest req)
|
|
88
|
+
{
|
|
89
|
+
if (req.Method == "__release" && req.Handle.HasValue)
|
|
90
|
+
{
|
|
91
|
+
s_handles.TryRemove(req.Handle.Value, out _);
|
|
92
|
+
return new InvokeResult(null, null);
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
// Return reflection metadata so JS can distinguish methods from properties.
|
|
96
|
+
if (req.Method == "__members__")
|
|
97
|
+
{
|
|
98
|
+
var t = req.Handle.HasValue
|
|
99
|
+
? (s_handles.TryGetValue(req.Handle.Value, out var h) ? h?.GetType() : null)
|
|
100
|
+
: ResolveType(req.Assembly, req.TypeName);
|
|
101
|
+
if (t is null) return new InvokeResult(null, $"Type not found: {req.TypeName}");
|
|
102
|
+
var inst = BindingFlags.Public | BindingFlags.Instance;
|
|
103
|
+
var stat = BindingFlags.Public | BindingFlags.Static;
|
|
104
|
+
var members = new
|
|
105
|
+
{
|
|
106
|
+
methods = t.GetMethods(inst).Where(m => !m.IsSpecialName).Select(m => m.Name).Distinct().ToArray(),
|
|
107
|
+
properties = t.GetProperties(inst).Select(p => p.Name).Distinct().ToArray(),
|
|
108
|
+
staticMethods = t.GetMethods(stat).Where(m => !m.IsSpecialName).Select(m => m.Name).Distinct().ToArray(),
|
|
109
|
+
staticProperties = t.GetProperties(stat).Select(p => p.Name).Distinct().ToArray(),
|
|
110
|
+
};
|
|
111
|
+
return new InvokeResult(JsonSerializer.SerializeToElement(members, JsonOptions.Default), null);
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
object? target = null;
|
|
115
|
+
Type? type = null;
|
|
116
|
+
|
|
117
|
+
if (req.Handle.HasValue)
|
|
118
|
+
{
|
|
119
|
+
if (!s_handles.TryGetValue(req.Handle.Value, out target))
|
|
120
|
+
return new InvokeResult(null, $"Invalid handle {req.Handle.Value}");
|
|
121
|
+
type = target?.GetType();
|
|
122
|
+
}
|
|
123
|
+
else
|
|
124
|
+
{
|
|
125
|
+
type = ResolveType(req.Assembly, req.TypeName);
|
|
126
|
+
if (type is null)
|
|
127
|
+
return new InvokeResult(null, $"Type not found: {req.TypeName}");
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
if (type is null)
|
|
131
|
+
return new InvokeResult(null, "Cannot determine target type");
|
|
132
|
+
|
|
133
|
+
var method = req.Method ?? throw new ArgumentException("Method is required");
|
|
134
|
+
|
|
135
|
+
// ── Constructor ──────────────────────────────────────────────────────
|
|
136
|
+
if (method == ".ctor")
|
|
137
|
+
{
|
|
138
|
+
var argElements = req.Args ?? [];
|
|
139
|
+
var ctors = type.GetConstructors(BindingFlags.Public | BindingFlags.Instance);
|
|
140
|
+
var ctor = ctors.FirstOrDefault(c => c.GetParameters().Length == argElements.Count)
|
|
141
|
+
?? ctors.FirstOrDefault();
|
|
142
|
+
if (ctor is null)
|
|
143
|
+
return new InvokeResult(null, $"No public constructor found on {type.FullName}");
|
|
144
|
+
var pars = ctor.GetParameters();
|
|
145
|
+
var ctorArgs = new object?[pars.Length];
|
|
146
|
+
for (int i = 0; i < pars.Length && i < argElements.Count; i++)
|
|
147
|
+
ctorArgs[i] = Coerce(argElements[i], pars[i].ParameterType);
|
|
148
|
+
return Box(ctor.Invoke(ctorArgs));
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
var isStatic = target is null;
|
|
152
|
+
var flags = (isStatic ? BindingFlags.Static : BindingFlags.Instance)
|
|
153
|
+
| BindingFlags.Public;
|
|
154
|
+
|
|
155
|
+
// ── Property getter / setter ─────────────────────────────────────────
|
|
156
|
+
if (method.StartsWith("get_", StringComparison.Ordinal))
|
|
157
|
+
{
|
|
158
|
+
var prop = GetCachedProperty(type, method[4..], flags);
|
|
159
|
+
if (prop is not null)
|
|
160
|
+
return Box(prop.GetValue(target));
|
|
161
|
+
}
|
|
162
|
+
if (method.StartsWith("set_", StringComparison.Ordinal) && req.Args?.Count == 1)
|
|
163
|
+
{
|
|
164
|
+
var prop = GetCachedProperty(type, method[4..], flags);
|
|
165
|
+
if (prop is not null)
|
|
166
|
+
{
|
|
167
|
+
prop.SetValue(target, Coerce(req.Args[0], prop.PropertyType));
|
|
168
|
+
return new InvokeResult(null, null);
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
// ── Regular method ───────────────────────────────────────────────────
|
|
173
|
+
var argElems = req.Args ?? [];
|
|
174
|
+
var mi = FindMethod(type, method, argElems.Count, flags);
|
|
175
|
+
if (mi is null)
|
|
176
|
+
return new InvokeResult(null,
|
|
177
|
+
$"Method '{method}' ({argElems.Count} args) not found on {type.FullName}");
|
|
178
|
+
|
|
179
|
+
var parameters = mi.GetParameters();
|
|
180
|
+
var callArgs = new object?[parameters.Length];
|
|
181
|
+
for (int i = 0; i < parameters.Length && i < argElems.Count; i++)
|
|
182
|
+
callArgs[i] = Coerce(argElems[i], parameters[i].ParameterType);
|
|
183
|
+
|
|
184
|
+
return Box(AwaitIfTask(mi.Invoke(target, callArgs)));
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
// ── helpers ───────────────────────────────────────────────────────────────
|
|
188
|
+
|
|
189
|
+
private static object? AwaitIfTask(object? value)
|
|
190
|
+
{
|
|
191
|
+
if (value is null) return null;
|
|
192
|
+
var t = value.GetType();
|
|
193
|
+
|
|
194
|
+
if (value is Task task)
|
|
195
|
+
{
|
|
196
|
+
task.Wait();
|
|
197
|
+
if (task.IsFaulted)
|
|
198
|
+
throw task.Exception!.InnerException ?? task.Exception;
|
|
199
|
+
return t.GetProperty("Result")?.GetValue(value);
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
if (t.IsGenericType && t.GetGenericTypeDefinition() == typeof(ValueTask<>))
|
|
203
|
+
{
|
|
204
|
+
var innerTask = (Task)t.GetMethod("AsTask")!.Invoke(value, null)!;
|
|
205
|
+
innerTask.Wait();
|
|
206
|
+
if (innerTask.IsFaulted)
|
|
207
|
+
throw innerTask.Exception!.InnerException ?? innerTask.Exception;
|
|
208
|
+
return innerTask.GetType().GetProperty("Result")?.GetValue(innerTask);
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
if (value is System.Threading.Tasks.ValueTask vt)
|
|
212
|
+
{
|
|
213
|
+
vt.AsTask().Wait();
|
|
214
|
+
return null;
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
return value;
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
private static Type? ResolveType(string? assemblyName, string? typeName)
|
|
221
|
+
{
|
|
222
|
+
if (string.IsNullOrEmpty(typeName)) return null;
|
|
223
|
+
var key = string.IsNullOrEmpty(assemblyName) ? typeName : $"{assemblyName}|{typeName}";
|
|
224
|
+
return s_typeCache.GetOrAdd(key, _ => ResolveTypeCore(assemblyName, typeName));
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
private static Type? ResolveTypeCore(string? assemblyName, string? typeName)
|
|
228
|
+
{
|
|
229
|
+
var fqn = string.IsNullOrEmpty(assemblyName) ? typeName! : $"{typeName}, {assemblyName}";
|
|
230
|
+
var t = Type.GetType(fqn);
|
|
231
|
+
if (t is not null) return t;
|
|
232
|
+
|
|
233
|
+
foreach (var asm in AppDomain.CurrentDomain.GetAssemblies())
|
|
234
|
+
{
|
|
235
|
+
t = asm.GetType(typeName!);
|
|
236
|
+
if (t is not null) return t;
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
if (!string.IsNullOrEmpty(assemblyName))
|
|
240
|
+
{
|
|
241
|
+
try
|
|
242
|
+
{
|
|
243
|
+
var asm = Assembly.Load(assemblyName);
|
|
244
|
+
t = asm.GetType(typeName!);
|
|
245
|
+
if (t is not null) return t;
|
|
246
|
+
}
|
|
247
|
+
catch { }
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
return null;
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
private static MethodInfo? FindMethod(Type type, string name, int argCount, BindingFlags flags)
|
|
254
|
+
{
|
|
255
|
+
var key = $"{type.FullName}|{name}|{argCount}|{(int)flags}";
|
|
256
|
+
return s_methodCache.GetOrAdd(key, _ => FindMethodCore(type, name, argCount, flags));
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
private static MethodInfo? FindMethodCore(Type type, string name, int argCount, BindingFlags flags)
|
|
260
|
+
{
|
|
261
|
+
var match = Array.Find(
|
|
262
|
+
type.GetMethods(flags),
|
|
263
|
+
m => m.Name == name && !m.IsGenericMethod && m.GetParameters().Length == argCount);
|
|
264
|
+
if (match is not null) return match;
|
|
265
|
+
return type.GetMethod(name, flags);
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
private static PropertyInfo? GetCachedProperty(Type type, string name, BindingFlags flags)
|
|
269
|
+
{
|
|
270
|
+
var key = $"{type.FullName}|{name}|{(int)flags}";
|
|
271
|
+
return s_propertyCache.GetOrAdd(key, _ => type.GetProperty(name, flags));
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
private static object? Coerce(JsonElement el, Type targetType)
|
|
275
|
+
{
|
|
276
|
+
if (el.ValueKind == JsonValueKind.Null) return null;
|
|
277
|
+
if (el.ValueKind == JsonValueKind.Object && el.TryGetProperty("__handle", out var h))
|
|
278
|
+
{
|
|
279
|
+
s_handles.TryGetValue(h.GetInt32(), out var obj);
|
|
280
|
+
return obj;
|
|
281
|
+
}
|
|
282
|
+
return JsonSerializer.Deserialize(el.GetRawText(), targetType, JsonOptions.Default);
|
|
283
|
+
}
|
|
284
|
+
|
|
285
|
+
private static InvokeResult Box(object? value)
|
|
286
|
+
{
|
|
287
|
+
if (value is null) return new InvokeResult(null, null);
|
|
288
|
+
|
|
289
|
+
var t = value.GetType();
|
|
290
|
+
|
|
291
|
+
if (t.IsPrimitive || t == typeof(string) || t == typeof(decimal)
|
|
292
|
+
|| t == typeof(DateTime) || t == typeof(DateTimeOffset)
|
|
293
|
+
|| t == typeof(TimeSpan) || t == typeof(Guid))
|
|
294
|
+
{
|
|
295
|
+
return new InvokeResult(
|
|
296
|
+
JsonSerializer.SerializeToElement(value, t, JsonOptions.Default), null);
|
|
297
|
+
}
|
|
298
|
+
|
|
299
|
+
// Arrays and IEnumerable — serialise as JSON arrays.
|
|
300
|
+
if (t.IsArray || (t != typeof(string) && value is IEnumerable enumerable))
|
|
301
|
+
{
|
|
302
|
+
try
|
|
303
|
+
{
|
|
304
|
+
var list = new List<object?>();
|
|
305
|
+
foreach (var item in (IEnumerable)value) list.Add(item);
|
|
306
|
+
return new InvokeResult(
|
|
307
|
+
JsonSerializer.SerializeToElement(list, typeof(List<object?>), JsonOptions.Default),
|
|
308
|
+
null);
|
|
309
|
+
}
|
|
310
|
+
catch { }
|
|
311
|
+
}
|
|
312
|
+
|
|
313
|
+
var id = Interlocked.Increment(ref s_nextHandle);
|
|
314
|
+
s_handles[id] = value;
|
|
315
|
+
return new InvokeResult(
|
|
316
|
+
JsonSerializer.SerializeToElement(
|
|
317
|
+
new { __handle = id, __type = t.FullName }, JsonOptions.Default),
|
|
318
|
+
null);
|
|
319
|
+
}
|
|
320
|
+
|
|
321
|
+
private static Exception Unwrap(Exception ex) =>
|
|
322
|
+
ex is TargetInvocationException { InnerException: { } inner } ? Unwrap(inner) : ex;
|
|
323
|
+
|
|
324
|
+
private static unsafe void WriteResponse(byte[] json, byte** responsePtr, int* responseLenPtr)
|
|
325
|
+
{
|
|
326
|
+
var ptr = (byte*)Marshal.AllocHGlobal(json.Length + 1);
|
|
327
|
+
Marshal.Copy(json, 0, (IntPtr)ptr, json.Length);
|
|
328
|
+
ptr[json.Length] = 0;
|
|
329
|
+
*responsePtr = ptr;
|
|
330
|
+
*responseLenPtr = json.Length;
|
|
331
|
+
}
|
|
332
|
+
}
|
|
333
|
+
|
|
334
|
+
// ── JSON types ────────────────────────────────────────────────────────────────
|
|
335
|
+
|
|
336
|
+
internal sealed record InvokeRequest(
|
|
337
|
+
string? Assembly,
|
|
338
|
+
string? TypeName,
|
|
339
|
+
string? Method,
|
|
340
|
+
int? Handle,
|
|
341
|
+
List<JsonElement>? Args
|
|
342
|
+
);
|
|
343
|
+
|
|
344
|
+
internal sealed record InvokeResult(JsonElement? Result, string? Error);
|
|
345
|
+
|
|
346
|
+
internal static class JsonOptions
|
|
347
|
+
{
|
|
348
|
+
internal static readonly JsonSerializerOptions Default = new()
|
|
349
|
+
{
|
|
350
|
+
PropertyNameCaseInsensitive = true,
|
|
351
|
+
PropertyNamingPolicy = JsonNamingPolicy.CamelCase,
|
|
352
|
+
};
|
|
353
|
+
}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
<Project Sdk="Microsoft.NET.Sdk">
|
|
2
|
+
|
|
3
|
+
<PropertyGroup>
|
|
4
|
+
<TargetFramework>net9.0</TargetFramework>
|
|
5
|
+
<AssemblyName>DotNetBridge</AssemblyName>
|
|
6
|
+
<RootNamespace>NativeScriptBridge</RootNamespace>
|
|
7
|
+
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
|
|
8
|
+
<Nullable>enable</Nullable>
|
|
9
|
+
<Optimize>true</Optimize>
|
|
10
|
+
<!-- Regular class library — hostfxr loads it at runtime; no NativeAOT needed. -->
|
|
11
|
+
<OutputType>Library</OutputType>
|
|
12
|
+
<!-- Required: dotnet publish on a class library won't emit runtimeconfig.json by default. -->
|
|
13
|
+
<GenerateRuntimeConfigurationFiles>true</GenerateRuntimeConfigurationFiles>
|
|
14
|
+
<!-- Allow hostfxr to load this assembly under net10+ without a re-publish.
|
|
15
|
+
Default (Minor) only accepts patch updates within major 9, which fails when the
|
|
16
|
+
host process is already running on net10. LatestMajor accepts any version >= 9. -->
|
|
17
|
+
<RollForward>LatestMajor</RollForward>
|
|
18
|
+
</PropertyGroup>
|
|
19
|
+
|
|
20
|
+
</Project>
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
|
Binary file
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@nativescript/windows",
|
|
3
|
-
"version": "0.1.0-alpha.
|
|
3
|
+
"version": "0.1.0-alpha.3",
|
|
4
4
|
"description": "NativeScript for using Windows v8",
|
|
5
5
|
"repository": {
|
|
6
6
|
"type": "git",
|
|
@@ -8,7 +8,8 @@
|
|
|
8
8
|
},
|
|
9
9
|
"scripts": {
|
|
10
10
|
"build": "pwsh -File build.ps1",
|
|
11
|
-
"build:x64": "pwsh -File build.ps1 -SkipArm64"
|
|
11
|
+
"build:x64": "pwsh -File build.ps1 -SkipArm64",
|
|
12
|
+
"build:nodotnet": "pwsh -File build.ps1 -SkipDotnet"
|
|
12
13
|
},
|
|
13
14
|
"keywords": ["nativescript", "windows", "winrt", "uwp"],
|
|
14
15
|
"license": "Apache-2.0",
|