Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/sh-akira/unitymemorymappedfile
共有メモリでUnityを外部アプリからコントロール
https://github.com/sh-akira/unitymemorymappedfile
Last synced: 3 months ago
JSON representation
共有メモリでUnityを外部アプリからコントロール
- Host: GitHub
- URL: https://github.com/sh-akira/unitymemorymappedfile
- Owner: sh-akira
- License: mit
- Created: 2019-01-14T07:18:19.000Z (almost 6 years ago)
- Default Branch: master
- Last Pushed: 2021-12-01T06:26:29.000Z (about 3 years ago)
- Last Synced: 2024-10-17T17:56:23.335Z (3 months ago)
- Language: C#
- Size: 42 KB
- Stars: 43
- Watchers: 3
- Forks: 4
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# UnityMemoryMappedFile
共有メモリでUnityを外部アプリからコントロール# 概要
UnityでWindowsの共有メモリ
MemoryMappedFile
を簡単に使用できるようにしたサンプルです。
UnityとWindowsアプリ間で通信ができます。
(ライブラリ自体はUnity以外同士の通信にも使用できます)
テスト環境は Unity 2018.1.6f1 で Scripting が .NET4.0 です# 更新履歴
2019/01/15
・初版# ビルド方法
UnityMemoryMappedFileWPF\UnityMemoryMappedFileWPF.slnを開いてリビルド
UnityMemoryMappedFile.dllがUnityMemoryMappedFileSample\Assetsに生成されるのを確認
UnityでUnityMemoryMappedFileSampleを開く
Playして、UnityMemoryMappedFileWPF側も実行する# 使用方法
サーバー(Unity)側:
``` csharp
using UnityEngine;
using UnityMemoryMappedFile;public class MemoryMappedFileController : MonoBehaviour
{
private MemoryMappedFileServer server;// Use this for initialization
void Start()
{
server = new MemoryMappedFileServer();
server.ReceivedEvent += Server_Received;
server.Start("SamplePipeName");}
private async void Server_Received(object sender, DataReceivedEventArgs e)
{
if (e.CommandType == typeof(PipeCommands.SendMessage))
{
var d = (PipeCommands.SendMessage)e.Data;
Debug.Log($"[Server]ReceiveFromClient:{d.Message}");
}
}// Update is called once per frame
async void Update()
{
if (Input.GetKeyDown(KeyCode.T))
{
await server.SendCommandAsync(new PipeCommands.SendMessage { Message = "TestFromServer" });
}
}
}
```クライアント(WPF)側:
``` csharp
using System.Windows;
using UnityMemoryMappedFile;namespace UnityMemoryMappedFileWPF
{
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}private MemoryMappedFileClient client;
private void Window_Loaded(object sender, RoutedEventArgs e)
{
client = new MemoryMappedFileClient();
client.ReceivedEvent += Client_Received;
client.Start("SamplePipeName");
}private async void SendButton_Click(object sender, RoutedEventArgs e)
{
await client.SendCommandAsync(new PipeCommands.SendMessage { Message = "TestFromWPF" });
}
private void Client_Received(object sender, DataReceivedEventArgs e)
{
if (e.CommandType == typeof(PipeCommands.SendMessage))
{
var d = (PipeCommands.SendMessage)e.Data;
MessageBox.Show($"[Client]ReceiveFromServer:{d.Message}");
}
}
}
}
```
Unity側でコマンドを受信した際にGameObjectに触るときはメインスレッドで実行する必要があります。
メインスレッドでActionを実行できるMainThreadInvokerも使用できます
``` csharp
[SerializeField]
private MainThreadInvoker mainThreadInvoker;
[SerializeField]
private Transform CubeTransform;private async void Server_Received(object sender, DataReceivedEventArgs e)
{
if (e.CommandType == typeof(PipeCommands.MoveObject))
{
var d = (PipeCommands.MoveObject)e.Data;
mainThreadInvoker.BeginInvoke(() => //別スレッドからGameObjectに触るときはメインスレッドで処理すること
{
var pos = CubeTransform.position;
pos.x += d.X;
CubeTransform.position = pos;
});
}
else if (e.CommandType == typeof(PipeCommands.GetCurrentPosition))
{
float x = 0.0f;
await mainThreadInvoker.InvokeAsync(() => x = CubeTransform.position.x); //GameObjectに触るときはメインスレッドで
await server.SendCommandAsync(new PipeCommands.ReturnCurrentPosition { CurrentX = x }, e.RequestId);
}
}
```
UnityMemoryMappedFile\PipeCommands.cs に追記することで自由にコマンドを増やすことができます。
アプリ間の通信時にはクラスごとバイナリにシリアライズされ転送されるため、
好きなデータをやり取りすることができます。
また、SendCommandWaitAsyncを使用することで、Unity側に値をリクエストして、その返答を受け取ることも可能です。
``` csharp
await client.SendCommandWaitAsync(new PipeCommands.GetCurrentPosition(), d =>
{
var ret = (PipeCommands.ReturnCurrentPosition)d;
Dispatcher.Invoke(() => ReceiveTextBlock.Text = $"{ret.CurrentX}");
});
```詳しい使用方法はソースをご確認ください。