WebAssembly 브라우저 앱 프로젝트와 JavaScript [JSImport]
/[JSExport]
interop
참고 항목
이 문서의 최신 버전은 아닙니다. 현재 릴리스는 이 문서의 .NET 9 버전을 참조 하세요.
Important
이 정보는 상업적으로 출시되기 전에 실질적으로 수정될 수 있는 시험판 제품과 관련이 있습니다. Microsoft는 여기에 제공된 정보에 대해 어떠한 명시적, 또는 묵시적인 보증을 하지 않습니다.
현재 릴리스는 이 문서의 .NET 9 버전을 참조 하세요.
이 문서에서는 interop을 사용하여[JSExport]
/[JSImport]
JS JavaScript(JS)에서 .NET을 실행하도록 WebAssembly 브라우저 앱 프로젝트를 설정하는 방법을 설명합니다. 자세한 내용 및 예제는 .NET WebAssembly의 JavaScript '[JSImport]'/'[JSExport]' interop을 참조하세요.
추가 지침은 .NET 런타임(dotnet/runtime
) GitHub 리포지토리에서 .NET WebAssembly 애플리케이션 구성 및 호스팅 지침을 참조하세요.
기존 JS 앱은 확장된 클라이언트 쪽 WebAssembly 지원을 사용하여 .NET 라이브러리 JS 를 다시 사용하거나 새 항목을 빌드할 수 있습니다. NET 기반 앱 및 프레임워크.
참고 항목
이 문서에서는 Blazor에 대한 종속성 없이 JS 앱에서 .NET을 실행하는 데 중점을 둡니다. 앱에서 Blazor WebAssembly interop를 사용하는 [JSImport]
/[JSExport]
방법에 대한 지침은 ASP.NET CoreBlazor와의 JavaScript JSImport/JSExport interop를 참조하세요.
이러한 방법은 WebAssembly(WASM)에서만 앱이 Blazor 실행되기를 기대하는 경우에 적합합니다. 라이브러리는 OperatingSystem.IsBrowser를 호출하여 앱이 WASM에서 실행 중 인지 확인하기 위해 런타임 검사를 수행할 수 있습니다.
필수 조건
wasm-tools
관련 MSBuild 대상을 가져오는 관리 명령 셸에 워크로드를 설치합니다.
dotnet workload install wasm-tools
Visual Studio 설치 관리자의 ASP.NET 및 웹 개발 워크로드에서 Visual Studio의 설치 관리자를 통해 도구를 설치할 수도 있습니다. 선택적 구성 요소 목록에서 .NET WebAssembly 빌드 도구 옵션을 선택합니다.
필요에 따라 다음 실험적 프로젝트 템플릿을 wasm-experimental
추가하는 워크로드를 설치합니다.
- 브라우저 앱 에서 WebAssembly에서 .NET을 시작하기 위한 WebAssembly 브라우저 앱입니다.
- 노드에서 시작하기 위한 WebAssembly 콘솔 앱 입니다.js-based console app.
워크로드를 설치한 후 새 프로젝트를 만들 때 이러한 새 템플릿을 선택할 수 있습니다. JS[JSImport]
/[JSExport]
interop을 기존 JS 앱에 통합하려는 경우에는 이 워크로드가 필요하지 않습니다.
dotnet workload install wasm-experimental
다음 명령을 사용하여 NuGet 패키지에서 Microsoft.NET.Runtime.WebAssembly.Templates
템플릿을 설치할 수도 있습니다.
dotnet new install Microsoft.NET.Runtime.WebAssembly.Templates
자세한 내용은 실험적 워크로드 및 프로젝트 템플릿 섹션을 참조하세요.
네임스페이스
이 문서에 설명된 JS interop API는 System.Runtime.InteropServices.JavaScript 네임스페이스의 특성에 의해 제어됩니다.
프로젝트 구성
JS interop을 사용하도록 프로젝트(.csproj
)를 구성하려면 다음을 수행합니다.
대상 프레임워크 모니커(
{TARGET FRAMEWORK}
자리 표시자)를 설정합니다.<TargetFramework>{TARGET FRAMEWORK}</TargetFramework>
.NET 7(
net7.0
) 이상이 지원됩니다.AllowUnsafeBlocks 속성을 사용하도록 설정하면 Roslyn 컴파일러의 코드 생성기가 JS interop에 대한 포인터를 사용할 수 있습니다.
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
Warning
JS interop API를 사용하려면 AllowUnsafeBlocks를 사용하도록 설정해야 합니다. 보안 및 안정성 위험을 초래할 수 있는 .NET 앱에서 안전하지 않은 코드를 구현할 때는 주의해야 합니다. 자세한 내용은 안전하지 않은 코드, 포인터 형식 및 함수 포인터를 참조하세요.
다음은 구성 후의 예제 프로젝트 파일(.csproj
)입니다. {TARGET FRAMEWORK}
자리 표시자는 대상 프레임워크입니다.
<Project Sdk="Microsoft.NET.Sdk.WebAssembly">
<PropertyGroup>
<TargetFramework>{TARGET FRAMEWORK}</TargetFramework>
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
</PropertyGroup>
</Project>
대상 프레임워크 모니커를 설정합니다.
<TargetFramework>net7.0</TargetFramework>
.NET 7(
net7.0
) 이상이 지원됩니다.런타임 식별자에 대해
browser-wasm
을 지정합니다.<RuntimeIdentifier>browser-wasm</RuntimeIdentifier>
실행 파일 출력 형식을 지정합니다.
<OutputType>Exe</OutputType>
AllowUnsafeBlocks 속성을 사용하도록 설정하면 Roslyn 컴파일러의 코드 생성기가 JS interop에 대한 포인터를 사용할 수 있습니다.
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
Warning
JS interop API를 사용하려면 AllowUnsafeBlocks를 사용하도록 설정해야 합니다. 보안 및 안정성 위험을 초래할 수 있는 .NET 앱에서 안전하지 않은 코드를 구현할 때는 주의해야 합니다. 자세한 내용은 안전하지 않은 코드, 포인터 형식 및 함수 포인터를 참조하세요.
WasmMainJSPath
을 디스크의 파일을 가리키도록 지정합니다. 이 파일은 앱과 함께 게시되지만 .NET을 기존 JS 앱에 통합하는 경우에는 파일을 사용할 필요가 없습니다.다음 예제에서 디스크의 JS 파일은
main.js
이지만 모든 JS 파일 이름은 허용됩니다.<WasmMainJSPath>main.js</WasmMainJSPath>
구성 후 예제 프로젝트 파일(.csproj
):
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net7.0</TargetFramework>
<RuntimeIdentifier>browser-wasm</RuntimeIdentifier>
<OutputType>Exe</OutputType>
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
<WasmMainJSPath>main.js</WasmMainJSPath>
<Nullable>enable</Nullable>
</PropertyGroup>
</Project>
WASM에서의 JavaScript interop
다음 예제의 API는 dotnet.js
에서 가져옵니다. 이러한 API를 사용하면 C# 코드로 가져올 수 있는 명명된 모듈을 설정하고 Program.Main
를 포함하여 .NET 코드에서 노출하는 메서드를 호출할 수 있습니다.
Important
이 문서 전체의 "가져오기" 및 "내보내기"는 .NET의 관점에서 정의됩니다.
- 앱은 .NET에서 호출할 수 있도록 JS 메서드를 가져옵니다.
- 앱은 JS에서 호출할 수 있도록 .NET 메서드를 내보냅니다.
다음 예제에서
dotnet.js
파일은 .NET WebAssembly 런타임을 만들고 시작하는 데 사용됩니다.dotnet.js
는 앱의 빌드 출력의 일부로 생성됩니다.Important
기존 앱과 통합하려면 게시 출력 폴더의 콘텐츠를 복사하고† 앱과 함께 제공될 수 있도록 기존 앱의 배포 자산에 rest 복사합니다. 프로덕션 배포의 경우 명령 셸에서 명령을 사용하여
dotnet publish -c Release
앱을 게시하고 앱과 함께 출력 폴더의 콘텐츠를 배포합니다.† 게시 출력 폴더는 게시 프로필의 대상 위치입니다. .NET 8 이상에서 프로필의 기본값 Release 은
bin/Release/{TARGET FRAMEWORK}/publish
자리 표시자가 대상 프레임워크(예:net8.0
)인 경우{TARGET FRAMEWORK}
입니다.dotnet.create()
는 .NET WebAssembly 런타임을 설정합니다.
setModuleImports
는 이름을 .NET으로 가져오기 위한 JS 함수 모듈과 연결합니다. 모듈에는 JS 현재 스톱워치 시간을 UI에 표시하는 데 사용하는 함수와 요소 선택기 및 시간이 포함dom.setInnerText
됩니다. 모듈의 이름은 모든 문자열일 수 있지만(파일 이름이 될 필요는 없음)JSImportAttribute
와 함께 사용되는 이름과 일치해야 합니다(이 문서의 뒷부분에서 설명).dom.setInnerText
함수는 C#으로 가져오고 C# 메서드SetInnerText
에 의해 호출됩니다.SetInnerText
메서드는 이 섹션의 뒷부분에 나와 있습니다.exports.StopwatchSample.Reset()
은StopwatchSample.Reset
에서 .NET(JS)을 호출합니다. C# 메서드는Reset
실행 중인 경우 stopwatch를 다시 시작하거나 실행되지 않는 경우 다시 설정합니다.Reset
메서드는 이 섹션의 뒷부분에 나와 있습니다.exports.StopwatchSample.Toggle()
은StopwatchSample.Toggle
에서 .NET(JS)을 호출합니다. C# 메서드는Toggle
현재 실행 중인지 여부에 따라 스톱워치를 시작하거나 중지합니다.Toggle
메서드는 이 섹션의 뒷부분에 나와 있습니다.runMain()
은Program.Main
를 실행합니다.
setModuleImports
는 이름을 .NET으로 가져오기 위한 JS 함수 모듈과 연결합니다. JS 모듈에는 현재 페이지 주소(URL)를 반환하는window.location.href
함수가 포함되어 있습니다. 모듈의 이름은 모든 문자열일 수 있지만(파일 이름이 될 필요는 없음)JSImportAttribute
와 함께 사용되는 이름과 일치해야 합니다(이 문서의 뒷부분에서 설명).window.location.href
함수는 C#으로 가져오고 C# 메서드GetHRef
에 의해 호출됩니다.GetHRef
메서드는 이 섹션의 뒷부분에 나와 있습니다.exports.MyClass.Greeting()
은MyClass.Greeting
에서 .NET(JS)을 호출합니다.Greeting
은 C# 메서드는window.location.href
함수 호출 결과를 포함하는 문자열을 반환합니다.Greeting
메서드는 이 섹션의 뒷부분에 나와 있습니다.dotnet.run()
은Program.Main
를 실행합니다.
JS 모듈:
import { dotnet } from './_framework/dotnet.js'
const { setModuleImports, getAssemblyExports, getConfig, runMain } = await dotnet
.withApplicationArguments("start")
.create();
setModuleImports('main.js', {
dom: {
setInnerText: (selector, time) =>
document.querySelector(selector).innerText = time
}
});
const config = getConfig();
const exports = await getAssemblyExports(config.mainAssemblyName);
document.getElementById('reset').addEventListener('click', e => {
exports.StopwatchSample.Reset();
e.preventDefault();
});
const pauseButton = document.getElementById('pause');
pauseButton.addEventListener('click', e => {
const isRunning = exports.StopwatchSample.Toggle();
pauseButton.innerText = isRunning ? 'Pause' : 'Start';
e.preventDefault();
});
await runMain();
import { dotnet } from './_framework/dotnet.js'
const { setModuleImports, getAssemblyExports, getConfig } = await dotnet
.withDiagnosticTracing(false)
.withApplicationArgumentsFromQuery()
.create();
setModuleImports('main.js', {
window: {
location: {
href: () => globalThis.window.location.href
}
}
});
const config = getConfig();
const exports = await getAssemblyExports(config.mainAssemblyName);
const text = exports.MyClass.Greeting();
console.log(text);
document.getElementById('out').innerHTML = text;
await dotnet.run();
import { dotnet } from './dotnet.js'
const is_browser = typeof window != "undefined";
if (!is_browser) throw new Error(`Expected to be running in a browser`);
const { setModuleImports, getAssemblyExports, getConfig } =
await dotnet.create();
setModuleImports("main.js", {
window: {
location: {
href: () => globalThis.window.location.href
}
}
});
const config = getConfig();
const exports = await getAssemblyExports(config.mainAssemblyName);
const text = exports.MyClass.Greeting();
console.log(text);
document.getElementById("out").innerHTML = text;
await dotnet.run();
C#에서 호출할 수 있도록 JS 함수를 가져오려면 일치하는 메서드 서명에서 새 JSImportAttribute를 사용합니다. JSImportAttribute의 첫 번째 매개 변수는 가져올 JS 함수의 이름이고 두 번째 매개 변수는 모듈의 이름입니다.
다음 예제에서는 SetInnerText
메서드가 호출되면 main.js
모듈에서 dom.setInnerText
함수가 호출됩니다.
[JSImport("dom.setInnerText", "main.js")]
internal static partial void SetInnerText(string selector, string content);
다음 예제에서는 GetHRef
메서드가 호출되면 main.js
모듈에서 window.location.href
함수가 호출됩니다.
[JSImport("window.location.href", "main.js")]
internal static partial string GetHRef();
가져온 메서드 서명에서 런타임에 의해 자동으로 마샬링되는 매개 변수 및 반환 값에 .NET 형식을 사용할 수 있습니다. JSMarshalAsAttribute<T>를 사용하여 가져온 메서드 매개 변수를 마샬링하는 방법을 제어합니다. 예를 들어, long
를 System.Runtime.InteropServices.JavaScript.JSType.Number 또는 System.Runtime.InteropServices.JavaScript.JSType.BigInt로 마샬링 하도록 선택할 수 있습니다. 호출 가능한 JS 함수로 마샬링되는 매개 변수로 Action/Func<TResult> 콜백을 전달할 수 있습니다. JS 및 관리형 개체 참조를 모두 전달할 수 있으며, 프록시 개체로 마샬링되어 프록시가 가비지 수집될 때까지 경계를 넘어 개체를 활성 상태로 유지합니다. JS 약속으로 마샬링되는 Task 결과를 사용하여 비동기 메서드를 가져오고 내보낼 수도 있습니다. 마샬링된 형식의 대부분은 가져온 메서드와 내보낸 메서드 모두에서 매개 변수 및 반환 값으로 양방향으로 작동합니다.
추가 형식 매핑 정보 및 예제는 .NET WebAssembly의 JavaScript '[JSImport]'/'[JSExport]' interop을 참조하세요.
전역 네임스페이스에서 액세스할 수 있는 함수는 함수 이름의 globalThis
접두사를 사용하고 모듈 이름을 제공하지 않고 [JSImport]
특성을 사용하여 가져올 수 있습니다. 다음 예제에서는 console.log
은 globalThis
로 접두사를 지정합니다. 가져온 함수는 C# 문자열 메시지(message
)를 수락하고 C# 문자열을 console.log
에 대해 JSString
에 마샬링하는 C# Log
메서드에 의해 호출됩니다.
[JSImport("globalThis.console.log")]
internal static partial void Log([JSMarshalAs<JSType.String>] string message);
JS에서 호출할 수 있도록 .NET 메서드를 내보내려면 JSExportAttribute를 사용합니다.
다음 예제에서는 각 메서드를 함수로 JS 내보내고 함수에서 JS 호출할 수 있습니다.
- 메서드는
Toggle
실행 상태에 따라 스톱워치를 시작하거나 중지합니다. - 이 메서드는
Reset
실행 중인 경우 stopwatch를 다시 시작하거나 실행되지 않는 경우 다시 설정합니다. - 이 메서드는
IsRunning
스톱워치가 실행 중인지를 나타냅니다.
[JSExport]
internal static bool Toggle()
{
if (stopwatch.IsRunning)
{
stopwatch.Stop();
return false;
}
else
{
stopwatch.Start();
return true;
}
}
[JSExport]
internal static void Reset()
{
if (stopwatch.IsRunning)
stopwatch.Restart();
else
stopwatch.Reset();
Render();
}
[JSExport]
internal static bool IsRunning() => stopwatch.IsRunning;
다음 예제에서 Greeting
메서드는 GetHRef
메서드를 호출한 결과를 포함하는 문자열을 반환합니다. 앞에서 설명한 것처럼, GetHref
C# 메서드는 main.js
모듈의 window.location.href
함수에 대해 JS을 호출합니다. window.location.href
는 현재 페이지 주소(URL)를 반환합니다.
[JSExport]
internal static string Greeting()
{
var text = $"Hello, World! Greetings from {GetHRef()}";
Console.WriteLine(text);
return text;
}
실험적 워크로드 및 프로젝트 템플릿
JS interop 기능을 시연하고 JS interop 프로젝트 템플릿을 가져오려면 wasm-experimental
워크로드를 설치합니다.
dotnet workload install wasm-experimental
wasm-experimental
워크로드에는 wasmbrowser
및 wasmconsole
의 두 가지 프로젝트 템플릿이 포함되어 있습니다. 이러한 템플릿은 현재 실험적입니다. 즉, 템플릿에 대한 개발자 워크플로는 진화하고 있습니다. 그러나 템플릿에 사용되는 .NET 및 JS API는 .NET 8에서 지원되며 WASM JS.
다음 명령을 사용하여 NuGet 패키지에서 Microsoft.NET.Runtime.WebAssembly.Templates
템플릿을 설치할 수도 있습니다.
dotnet new install Microsoft.NET.Runtime.WebAssembly.Templates
브라우저 앱
명령줄에서 템플릿을 wasmbrowser
사용하여 브라우저 앱을 만들 수 있습니다. 이 웹앱은 .NET을 JS 사용하여 브라우저에서 함께 사용하는 방법을 보여 주는 웹앱을 만듭니다.
dotnet new wasmbrowser
또는 Visual Studio에서 프로젝트 템플릿을 사용하여 WebAssembly Browser App 앱을 만들 수 있습니다.
Visual Studio에서 앱을 빌드하거나 .NET CLI를 사용하여 앱을 빌드합니다.
dotnet build
Visual Studio 또는 .NET CLI를 사용하여 앱을 빌드하고 실행합니다.
dotnet run
또는 다음 명령을 설치하고 사용합니다.dotnet serve
dotnet serve -d:bin/$(Configuration)/{TARGET FRAMEWORK}/publish
앞의 예제 {TARGET FRAMEWORK}
에서 자리 표시자는 대상 프레임워크 모니커입니다.
노드.js 콘솔 앱
템플릿을 사용하여 콘솔 앱을 wasmconsole
만들 수 있습니다. 그러면 노드 또는 V8 콘솔 앱으로 WASM 실행되는 앱이 만들어집니다.js
dotnet new wasmconsole
또는 Visual Studio에서 프로젝트 템플릿을 사용하여 WebAssembly Console App 앱을 만들 수 있습니다.
Visual Studio에서 앱을 빌드하거나 .NET CLI를 사용하여 앱을 빌드합니다.
dotnet build
Visual Studio 또는 .NET CLI를 사용하여 앱을 빌드하고 실행합니다.
dotnet run
또는 파일이 포함된 게시 출력 디렉터리에서 정적 파일 서버를 시작합니다.main.mjs
node bin/$(Configuration)/{TARGET FRAMEWORK}/{PATH}/main.mjs
앞의 예제 {TARGET FRAMEWORK}
에서 자리 표시자는 대상 프레임워크 모니커이고 {PATH}
자리 표시자는 파일의 경로 main.mjs
입니다.
추가 리소스
- .NET WebAssembly의 JavaScript '[JSImport]'/'[JSExport]' interop
- ASP.NET Core와 JavaScript JSImport/JSExport interop Blazor
- API 설명서
dotnet/runtime
GitHub 리포지토리에서:- .NET 7의 JavaScript 앱에서 .NET 사용
ASP.NET Core