Unity【Live Capture】- 关于人脸捕捉的解决方案(一)

举报
CoderZ1010 发表于 2022/09/25 04:27:32 2022/09/25
【摘要】 最近项目里有人脸捕捉的需求,刚开始时参考的下面这篇文章,使用官方发布的Facial AR Remote,需要我们自己构建IOS客户端,因此需要准备包括MacOS操作系统、Xcode等开发环境,在Unity构建出Xcode工程后,还要考虑开发许可证等问题,而且在尝试时,我使用的Xcode13版本,在编译上还有一些问题,比较麻烦。 htt...

最近项目里有人脸捕捉的需求,刚开始时参考的下面这篇文章,使用官方发布的Facial AR Remote,需要我们自己构建IOS客户端,因此需要准备包括MacOS操作系统、Xcode等开发环境,在Unity构建出Xcode工程后,还要考虑开发许可证等问题,而且在尝试时,我使用的Xcode13版本,在编译上还有一些问题,比较麻烦。

https://www.163.com/dy/article/E70U8CLT0526E124.html

随后发现了另一个解决方案,即Live Capture,IOS客户端已经发布于App Store中,名称Unity Face Capture:

Live Capture在Package Manager中通过git url的方式进行添加,地址:

http://com.unity.live-capture

Live Capture官方手册地址:

https://docs.unity.cn/Packages/com.unity.live-capture@1.0/manual/index.html

PDF文档下载地址:

https://link.csdn.net/?target=https%3A%2F%2Fforum.unity.com%2Fattachments%2Flive-capture-apps-startup-guide-pdf.961348%2F

文档很详细,也可以参考下面的步骤:

1.新建空物体,添加Take Recoder组件,它依赖Playable Director组件,添加Take Recorder时自动添加Playable Director:

2.将制作好的包含人脸的模型拖入场景,挂载ARKit Face Actor组件,然后将其做成Prefab预制体。

3.在Take Recoder组件下方点击+按钮,添加ARKit Face Device,并将其Actor设为步骤2中挂载了ARKit Face Actor组件的人脸模型:

4.在Project窗口右键,Create / Live Capture / ARKit Face Capture / Mapper,创建一个Head Mapper资产,将其Rig Prefab设为步骤2中生成的Prefab,并设置人脸模型中对应的的Left Eye、RightEye、Head骨骼节点:

5.点击Head Mapper资产中的Add Renderer按钮,绑定BlendShape,名称如果与ARKit中要求一致则会自动绑定好,否则需要一一对应设置:

6.将编辑好的Head Mapper资产赋值给步骤2挂载的ARKit Face Actor组件中的Mapper:

7.Window / Live Capture / Connections 打开Connections窗口,创建服务器,点击Start即可启动:

8.启动后打开IOS客户端,点击Connect进行连接,若连接不上,检查一下手机和电脑是否处于同一网段,检查电脑防火墙,最好都关掉。

另外值得注意的是,服务端的启动是通过在Connections编辑器窗口中点击Start开启的,因此它的使用环境是在Unity编辑器环境中,如果想打包后运行时使用,需要将其从Package Manager中迁移到工程Assets目录下,并创建脚本编写启动方法:

创建LiveCaptureServer类,继承Server:


  
  1. using System;
  2. using System.IO;
  3. using System.Linq;
  4. using System.Text;
  5. using UnityEngine;
  6. using System.Collections.Generic;
  7. using Unity.LiveCapture.Networking;
  8. using Unity.LiveCapture.Networking.Discovery;
  9. namespace Unity.LiveCapture.CompanionApp
  10. {
  11. [CreateAssetMenu(menuName = "Live Capture/Server")]
  12. public class LiveCaptureServer : Server
  13. {
  14. const int k_DefaultPort = 9000;
  15. /// <summary>
  16. /// The server executes this event when a client has connected.
  17. /// </summary>
  18. public static event Action<ICompanionAppClient> ClientConnected = delegate { };
  19. /// <summary>
  20. /// The server executes this event when a client has disconnected.
  21. /// </summary>
  22. public static event Action<ICompanionAppClient> ClientDisconnected = delegate { };
  23. struct ConnectHandler
  24. {
  25. public string Name;
  26. public DateTime Time;
  27. public Func<ICompanionAppClient, bool> Handler;
  28. }
  29. readonly Dictionary<string, Type> s_TypeToClientType = new Dictionary<string, Type>();
  30. static readonly List<ConnectHandler> s_ClientConnectHandlers = new List<ConnectHandler>();
  31. /// <summary>
  32. /// Adds a callback used to take ownership of a client that has connected.
  33. /// </summary>
  34. /// <param name="handler">The callback function. It must return true if it takes ownership of a client.</param>
  35. /// <param name="name">The name of the client to prefer. If set, this handler has priority over clients that have the given name.</param>
  36. /// <param name="time">The time used to determine the priority of handlers when many are listening for the same
  37. /// client <paramref name="name"/>. More recent values have higher priority.</param>
  38. /// <exception cref="ArgumentNullException">Thrown if <paramref name="handler"/> is null.</exception>
  39. public static void RegisterClientConnectHandler(Func<ICompanionAppClient, bool> handler, string name, DateTime time)
  40. {
  41. if (handler == null)
  42. throw new ArgumentNullException(nameof(handler));
  43. DeregisterClientConnectHandler(handler);
  44. s_ClientConnectHandlers.Add(new ConnectHandler
  45. {
  46. Name = name,
  47. Time = time,
  48. Handler = handler,
  49. });
  50. }
  51. /// <summary>
  52. /// Removes a client connection callback.
  53. /// </summary>
  54. /// <param name="handler">The callback to remove.</param>>
  55. /// <exception cref="ArgumentNullException">Thrown if <paramref name="handler"/> is null.</exception>
  56. public static void DeregisterClientConnectHandler(Func<ICompanionAppClient, bool> handler)
  57. {
  58. if (handler == null)
  59. throw new ArgumentNullException(nameof(handler));
  60. for (var i = 0; i < s_ClientConnectHandlers.Count; i++)
  61. {
  62. if (s_ClientConnectHandlers[i].Handler == handler)
  63. {
  64. s_ClientConnectHandlers.RemoveAt(i);
  65. }
  66. }
  67. }
  68. public void Init()
  69. {
  70. foreach (var (type, attributes) in AttributeUtility.GetAllTypes<ClientAttribute>())
  71. {
  72. if (!typeof(CompanionAppClient).IsAssignableFrom(type))
  73. {
  74. Debug.LogError($"{type.FullName} must be assignable from {nameof(CompanionAppClient)} to use the {nameof(ClientAttribute)} attribute.");
  75. continue;
  76. }
  77. foreach (var attribute in attributes)
  78. {
  79. s_TypeToClientType[attribute.Type] = type;
  80. }
  81. }
  82. }
  83. [SerializeField, Tooltip("The TCP port on which the server will listen for incoming connections. Changes to the port only take effect after restarting the server.")]
  84. int m_Port = k_DefaultPort;
  85. [SerializeField, Tooltip("Start the server automatically after entering play mode.")]
  86. bool m_AutoStartOnPlay = true;
  87. readonly DiscoveryServer m_Discovery = new DiscoveryServer();
  88. readonly NetworkServer m_Server = new NetworkServer();
  89. readonly Dictionary<Remote, ICompanionAppClient> m_RemoteToClient = new Dictionary<Remote, ICompanionAppClient>();
  90. /// <summary>
  91. /// The TCP port on which the server will listen for incoming connections.
  92. /// </summary>
  93. /// <remarks>
  94. /// Changes to the port only take effect after restarting the server.
  95. /// </remarks>
  96. public int Port
  97. {
  98. get => m_Port;
  99. set
  100. {
  101. if (m_Port != value)
  102. {
  103. m_Port = value;
  104. OnServerChanged(true);
  105. }
  106. }
  107. }
  108. /// <summary>
  109. /// Start the server automatically after entering play mode.
  110. /// </summary>
  111. public bool AutoStartOnPlay
  112. {
  113. get => m_AutoStartOnPlay;
  114. set
  115. {
  116. if (m_AutoStartOnPlay != value)
  117. {
  118. m_AutoStartOnPlay = value;
  119. OnServerChanged(true);
  120. }
  121. }
  122. }
  123. /// <summary>
  124. /// Are clients able to connect to the server.
  125. /// </summary>
  126. public bool IsRunning => m_Server.IsRunning;
  127. /// <summary>
  128. /// The number of clients currently connected to the server.
  129. /// </summary>
  130. public int ClientCount => m_RemoteToClient.Count;
  131. /// <inheritdoc/>
  132. protected override void OnEnable()
  133. {
  134. base.OnEnable();
  135. m_Server.RemoteConnected += OnClientConnected;
  136. m_Server.RemoteDisconnected += OnClientDisconnected;
  137. }
  138. /// <inheritdoc/>
  139. protected override void OnDisable()
  140. {
  141. base.OnDisable();
  142. m_Discovery.Stop();
  143. m_Server.Stop();
  144. m_Server.RemoteConnected -= OnClientConnected;
  145. m_Server.RemoteDisconnected -= OnClientDisconnected;
  146. }
  147. /// <summary>
  148. /// Gets the currently connected clients.
  149. /// </summary>
  150. /// <returns>A new collection containing the client handles.</returns>
  151. public IEnumerable<ICompanionAppClient> GetClients()
  152. {
  153. return m_RemoteToClient.Values;
  154. }
  155. /// <inheritdoc />
  156. public override string GetName() => "Companion App Server";
  157. /// <summary>
  158. /// Start listening for clients connections.
  159. /// </summary>
  160. public void StartServer()
  161. {
  162. if (!NetworkUtilities.IsPortAvailable(m_Port))
  163. {
  164. Debug.LogError($"Unable to start server: Port {m_Port} is in use by another program! Close the other program, or assign a free port using the Live Capture Window.");
  165. return;
  166. }
  167. if (m_Server.StartServer(m_Port))
  168. {
  169. // start server discovery
  170. var config = new ServerData(
  171. "Live Capture",
  172. Environment.MachineName,
  173. m_Server.ID,
  174. PackageUtility.GetVersion(LiveCaptureInfo.Version)
  175. );
  176. var endPoints = m_Server.EndPoints.ToArray();
  177. m_Discovery.Start(config, endPoints);
  178. }
  179. OnServerChanged(false);
  180. }
  181. /// <summary>
  182. /// Disconnects all clients and stop listening for new connections.
  183. /// </summary>
  184. public void StopServer()
  185. {
  186. m_Server.Stop();
  187. m_Discovery.Stop();
  188. OnServerChanged(false);
  189. }
  190. /// <inheritdoc/>
  191. public override void OnUpdate()
  192. {
  193. m_Server.Update();
  194. m_Discovery.Update();
  195. }
  196. void OnClientConnected(Remote remote)
  197. {
  198. m_Server.RegisterMessageHandler(remote, InitializeClient, false);
  199. }
  200. void OnClientDisconnected(Remote remote, DisconnectStatus status)
  201. {
  202. if (m_RemoteToClient.TryGetValue(remote, out var client))
  203. {
  204. try
  205. {
  206. ClientDisconnected.Invoke(client);
  207. }
  208. catch (Exception e)
  209. {
  210. Debug.LogError(e);
  211. }
  212. m_RemoteToClient.Remove(remote);
  213. OnServerChanged(false);
  214. }
  215. }
  216. void InitializeClient(Message message)
  217. {
  218. try
  219. {
  220. if (message.ChannelType != ChannelType.ReliableOrdered)
  221. {
  222. return;
  223. }
  224. var streamReader = new StreamReader(message.Data, Encoding.UTF8);
  225. var json = streamReader.ReadToEnd();
  226. var data = default(ClientInitialization);
  227. try
  228. {
  229. data = JsonUtility.FromJson<ClientInitialization>(json);
  230. }
  231. catch (Exception)
  232. {
  233. Debug.LogError($"{nameof(CompanionAppServer)} failed to initialize client connection! Could not parse JSON: {json}");
  234. return;
  235. }
  236. if (!s_TypeToClientType.TryGetValue(data.Type, out var clientType))
  237. {
  238. Debug.LogError($"Unknown client type \"{data.Type}\" connected to {nameof(CompanionAppServer)}!");
  239. return;
  240. }
  241. var remote = message.Remote;
  242. var client = Activator.CreateInstance(clientType, m_Server, remote, data) as CompanionAppClient;
  243. client.SendProtocol();
  244. m_RemoteToClient.Add(remote, client);
  245. AssignOwner(client);
  246. ClientConnected.Invoke(client);
  247. OnServerChanged(false);
  248. }
  249. catch (Exception e)
  250. {
  251. Debug.LogException(e);
  252. }
  253. finally
  254. {
  255. message.Dispose();
  256. }
  257. }
  258. void AssignOwner(ICompanionAppClient client)
  259. {
  260. // connect to the registered handler that was most recently used with this client if possible
  261. foreach (var handler in s_ClientConnectHandlers.OrderByDescending(h => h.Time.Ticks))
  262. {
  263. try
  264. {
  265. if (handler.Name == client.Name)
  266. {
  267. if (handler.Handler(client))
  268. return;
  269. }
  270. }
  271. catch (Exception e)
  272. {
  273. Debug.LogException(e);
  274. }
  275. }
  276. // fall back to the first free device that is compatible with the client
  277. foreach (var handler in s_ClientConnectHandlers)
  278. {
  279. try
  280. {
  281. if (handler.Handler(client))
  282. return;
  283. }
  284. catch (Exception e)
  285. {
  286. Debug.LogException(e);
  287. }
  288. }
  289. }
  290. }
  291. }

 更改CompanionAppDevice类:


  
  1. using System;
  2. using UnityEngine;
  3. namespace Unity.LiveCapture.CompanionApp
  4. {
  5. /// <summary>
  6. /// A type of <see cref="LiveCaptureDevice"/> that uses a <see cref="ICompanionAppClient"/> for communication.
  7. /// </summary>
  8. interface ICompanionAppDevice
  9. {
  10. /// <summary>
  11. /// Clears the client assigned to this device.
  12. /// </summary>
  13. void ClearClient();
  14. }
  15. /// <summary>
  16. /// A type of <see cref="LiveCaptureDevice"/> that uses a <see cref="ICompanionAppClient"/> for communication.
  17. /// </summary>
  18. /// <typeparam name="TClient">The type of client this device communicates with.</typeparam>
  19. public abstract class CompanionAppDevice<TClient> : LiveCaptureDevice, ICompanionAppDevice
  20. where TClient : class, ICompanionAppClient
  21. {
  22. bool m_ClientRegistered;
  23. bool m_Recording;
  24. TClient m_Client;
  25. readonly SlateChangeTracker m_SlateChangeTracker = new SlateChangeTracker();
  26. readonly TakeNameFormatter m_TakeNameFormatter = new TakeNameFormatter();
  27. string m_LastAssetName;
  28. bool TryGetInternalClient(out ICompanionAppClientInternal client)
  29. {
  30. client = m_Client as ICompanionAppClientInternal;
  31. return client != null;
  32. }
  33. /// <summary>
  34. /// This function is called when the object becomes enabled and active.
  35. /// </summary>
  36. protected virtual void OnEnable()
  37. {
  38. CompanionAppServer.ClientDisconnected += OnClientDisconnected;
  39. LiveCaptureServer.ClientDisconnected += OnClientDisconnected;
  40. RegisterClient();
  41. }
  42. /// <summary>
  43. /// This function is called when the behaviour becomes disabled.
  44. /// </summary>
  45. /// <remaks>
  46. /// This is also called when the object is destroyed and can be used for any cleanup code.
  47. /// When scripts are reloaded after compilation has finished, OnDisable will be called, followed by an OnEnable after the script has been loaded.
  48. /// </remaks>
  49. protected virtual void OnDisable()
  50. {
  51. CompanionAppServer.ClientDisconnected -= OnClientDisconnected;
  52. CompanionAppServer.DeregisterClientConnectHandler(OnClientConnected);
  53. LiveCaptureServer.ClientConnected -= OnClientDisconnected;
  54. LiveCaptureServer.DeregisterClientConnectHandler(OnClientConnected);
  55. StopRecording();
  56. UnregisterClient();
  57. }
  58. /// <summary>
  59. /// This function is called when the behaviour gets destroyed.
  60. /// </summary>
  61. protected override void OnDestroy()
  62. {
  63. base.OnDestroy();
  64. ClearClient();
  65. }
  66. /// <inheritdoc/>
  67. public override bool IsReady()
  68. {
  69. return m_Client != null;
  70. }
  71. /// <inheritdoc/>
  72. public override bool IsRecording()
  73. {
  74. return m_Recording;
  75. }
  76. /// <inheritdoc/>
  77. public override void StartRecording()
  78. {
  79. if (!m_Recording)
  80. {
  81. m_Recording = true;
  82. OnRecordingChanged();
  83. SendRecordingState();
  84. }
  85. }
  86. /// <inheritdoc/>
  87. public override void StopRecording()
  88. {
  89. if (m_Recording)
  90. {
  91. m_Recording = false;
  92. OnRecordingChanged();
  93. SendRecordingState();
  94. }
  95. }
  96. /// <summary>
  97. /// Gets the client currently assigned to this device.
  98. /// </summary>
  99. /// <returns>The assigned client, or null if none is assigned.</returns>
  100. public TClient GetClient()
  101. {
  102. return m_Client;
  103. }
  104. /// <summary>
  105. /// Assigns a client to this device.
  106. /// </summary>
  107. /// <param name="client">The client to assign, or null to clear the assigned client.</param>
  108. /// <param name="rememberAssignment">Try to auto-assign the client to this device when it reconnects in the future.</param>
  109. public void SetClient(TClient client, bool rememberAssignment)
  110. {
  111. if (m_Client != client)
  112. {
  113. UnregisterClient();
  114. if (m_Client != null)
  115. {
  116. ClientMappingDatabase.DeregisterClientAssociation(this, m_Client, rememberAssignment);
  117. }
  118. m_Client = client;
  119. if (m_Client != null)
  120. {
  121. // if any device is also using this client, we must clear the client from the previous device.
  122. if (ClientMappingDatabase.TryGetDevice(client, out var previousDevice))
  123. {
  124. previousDevice.ClearClient();
  125. }
  126. ClientMappingDatabase.RegisterClientAssociation(this, m_Client, rememberAssignment);
  127. }
  128. RegisterClient();
  129. }
  130. }
  131. void RegisterClient()
  132. {
  133. if (!isActiveAndEnabled ||  m_ClientRegistered)
  134. {
  135. return;
  136. }
  137. LiveCaptureServer.DeregisterClientConnectHandler(OnClientConnected);
  138. CompanionAppServer.DeregisterClientConnectHandler(OnClientConnected);
  139. m_SlateChangeTracker.Reset();
  140. if (TryGetInternalClient(out var client))
  141. {
  142. client.SetDeviceMode += ClientSetDeviceMode;
  143. client.StartRecording += ClientStartRecording;
  144. client.StopRecording += ClientStopRecording;
  145. client.StartPlayer += ClientStartPlayer;
  146. client.StopPlayer += ClientStopPlayer;
  147. client.PausePlayer += ClientPausePlayer;
  148. client.SetPlayerTime += ClientSetPlayerTime;
  149. client.SetSelectedTake += ClientSetSelectedTake;
  150. client.SetTakeData += ClientSetTakeData;
  151. client.DeleteTake += ClientDeleteTake;
  152. client.SetIterationBase += ClientSetIterationBase;
  153. client.ClearIterationBase += ClientClearIterationBase;
  154. client.TexturePreviewRequested += OnTexturePreviewRequested;
  155. OnClientAssigned();
  156. client.SendInitialize();
  157. UpdateClient();
  158.  m_ClientRegistered = true;
  159. }
  160. else
  161. {
  162. ClientMappingDatabase.TryGetClientAssignment(this, out var clientName, out var time);
  163. LiveCaptureServer.RegisterClientConnectHandler(OnClientConnected, clientName, time);
  164. CompanionAppServer.RegisterClientConnectHandler(OnClientConnected, clientName, time);
  165. }
  166. }
  167. void UnregisterClient()
  168. {
  169. if (!m_ClientRegistered)
  170. {
  171. return;
  172. }
  173. if (TryGetInternalClient(out var client))
  174. {
  175. OnClientUnassigned();
  176. client.SendEndSession();
  177. client.SetDeviceMode -= ClientSetDeviceMode;
  178. client.StartRecording -= ClientStartRecording;
  179. client.StopRecording -= ClientStopRecording;
  180. client.StartPlayer -= ClientStartPlayer;
  181. client.StopPlayer -= ClientStopPlayer;
  182. client.PausePlayer -= ClientPausePlayer;
  183. client.SetPlayerTime -= ClientSetPlayerTime;
  184. client.SetSelectedTake -= ClientSetSelectedTake;
  185. client.SetTakeData -= ClientSetTakeData;
  186. client.DeleteTake -= ClientDeleteTake;
  187. client.SetIterationBase -= ClientSetIterationBase;
  188. client.ClearIterationBase -= ClientClearIterationBase;
  189. client.TexturePreviewRequested -= OnTexturePreviewRequested;
  190. m_ClientRegistered = false;
  191. }
  192. }
  193. /// <inheritdoc />
  194. public void ClearClient()
  195. {
  196. SetClient(null, true);
  197. }
  198. /// <summary>
  199. /// Called to send the device state to the client.
  200. /// </summary>
  201. public virtual void UpdateClient()
  202. {
  203. var takeRecorder = GetTakeRecorder();
  204. if (takeRecorder != null)
  205. {
  206. SendDeviceState(takeRecorder.IsLive());
  207. var slate = takeRecorder.GetActiveSlate();
  208. var hasSlate = slate != null;
  209. var slateChanged = m_SlateChangeTracker.Changed(slate);
  210. var take = hasSlate ? slate.Take : null;
  211. var assetName = GetAssetName();
  212. var assetNameChanged = assetName != m_LastAssetName;
  213. m_LastAssetName = assetName;
  214. if (TryGetInternalClient(out var client))
  215. {
  216. client.SendFrameRate(takeRecorder.IsLive() || take == null ? takeRecorder.FrameRate : take.FrameRate);
  217. client.SendHasSlate(hasSlate);
  218. client.SendSlateDuration(hasSlate ? slate.Duration : 0d);
  219. client.SendSlateIsPreviewing(takeRecorder.IsPreviewPlaying());
  220. client.SendSlatePreviewTime(takeRecorder.GetPreviewTime());
  221. if (slateChanged || assetNameChanged)
  222. {
  223. if (hasSlate)
  224. m_TakeNameFormatter.ConfigureTake(slate.SceneNumber, slate.ShotName, slate.TakeNumber);
  225. else
  226. m_TakeNameFormatter.ConfigureTake(0, "Shot", 0);
  227. client.SendNextTakeName(m_TakeNameFormatter.GetTakeName());
  228. client.SendNextAssetName(m_TakeNameFormatter.GetAssetName());
  229. }
  230. }
  231. if (slateChanged)
  232. {
  233. SendSlateDescriptor(slate);
  234. }
  235. }
  236. SendRecordingState();
  237. }
  238. /// <summary>
  239. /// Gets the name used for the take asset name.
  240. /// </summary>
  241. /// <returns>The name of the asset.</returns>
  242. protected virtual string GetAssetName() { return name; }
  243. /// <summary>
  244. /// The device calls this method when a new client is assigned.
  245. /// </summary>
  246. protected virtual void OnClientAssigned() {}
  247. /// <summary>
  248. /// The device calls this method when the client is unassigned.
  249. /// </summary>
  250. protected virtual void OnClientUnassigned() {}
  251. /// <summary>
  252. /// The device calls this method when the recording state has changed.
  253. /// </summary>
  254. protected virtual void OnRecordingChanged() {}
  255. /// <summary>
  256. /// The device calls this method when the slate has changed.
  257. /// </summary>
  258. protected virtual void OnSlateChanged(ISlate slate) {}
  259. void ClientStartRecording()
  260. {
  261. var takeRecorder = GetTakeRecorder();
  262. if (takeRecorder != null)
  263. {
  264. takeRecorder.StartRecording();
  265. }
  266. Refresh();
  267. }
  268. void ClientStopRecording()
  269. {
  270. var takeRecorder = GetTakeRecorder();
  271. if (takeRecorder != null)
  272. {
  273. takeRecorder.StopRecording();
  274. }
  275. Refresh();
  276. }
  277. void ClientSetDeviceMode(DeviceMode deviceMode)
  278. {
  279. var takeRecorder = GetTakeRecorder();
  280. if (takeRecorder != null)
  281. {
  282. takeRecorder.SetLive(deviceMode == DeviceMode.LiveStream);
  283. SendDeviceState(takeRecorder.IsLive());
  284. }
  285. }
  286. void ClientStartPlayer()
  287. {
  288. var takeRecorder = GetTakeRecorder();
  289. if (takeRecorder != null)
  290. {
  291. takeRecorder.PlayPreview();
  292. }
  293. Refresh();
  294. }
  295. void ClientStopPlayer()
  296. {
  297. var takeRecorder = GetTakeRecorder();
  298. if (takeRecorder != null)
  299. {
  300. takeRecorder.PausePreview();
  301. takeRecorder.SetPreviewTime(0d);
  302. }
  303. Refresh();
  304. }
  305. void ClientPausePlayer()
  306. {
  307. var takeRecorder = GetTakeRecorder();
  308. if (takeRecorder != null)
  309. {
  310. takeRecorder.PausePreview();
  311. }
  312. Refresh();
  313. }
  314. void ClientSetPlayerTime(double time)
  315. {
  316. var takeRecorder = GetTakeRecorder();
  317. if (takeRecorder != null)
  318. {
  319. takeRecorder.SetPreviewTime(time);
  320. }
  321. Refresh();
  322. }
  323. void SendDeviceState()
  324. {
  325. var takeRecorder = GetTakeRecorder();
  326. if (takeRecorder != null)
  327. {
  328. SendDeviceState(takeRecorder.IsLive());
  329. }
  330. }
  331. void SendDeviceState(bool isLive)
  332. {
  333. if (TryGetInternalClient(out var client))
  334. {
  335. client.SendDeviceMode(isLive ? DeviceMode.LiveStream : DeviceMode.Playback);
  336. }
  337. }
  338. void SendRecordingState()
  339. {
  340. if (TryGetInternalClient(out var client))
  341. {
  342. client.SendRecordingState(IsRecording());
  343. }
  344. }
  345. void SendSlateDescriptor()
  346. {
  347. var takeRecorder = GetTakeRecorder();
  348. if (takeRecorder != null)
  349. {
  350. SendSlateDescriptor(takeRecorder.GetActiveSlate());
  351. }
  352. }
  353. void SendSlateDescriptor(ISlate slate)
  354. {
  355. if (TryGetInternalClient(out var client))
  356. {
  357. client.SendSlateDescriptor(SlateDescriptor.Create(slate));
  358. }
  359. OnSlateChanged(slate);
  360. }
  361. void ClientSetSelectedTake(Guid guid)
  362. {
  363. var takeRecorder = GetTakeRecorder();
  364. if (takeRecorder != null)
  365. {
  366. TakeManager.Default.SelectTake(takeRecorder.GetActiveSlate(), guid);
  367. SendSlateDescriptor();
  368. Refresh();
  369. }
  370. }
  371. void ClientSetTakeData(TakeDescriptor descriptor)
  372. {
  373. TakeManager.Default.SetTakeData(descriptor);
  374. SendSlateDescriptor();
  375. Refresh();
  376. }
  377. void ClientDeleteTake(Guid guid)
  378. {
  379. TakeManager.Default.DeleteTake(guid);
  380. SendSlateDescriptor();
  381. Refresh();
  382. }
  383. void ClientSetIterationBase(Guid guid)
  384. {
  385. var takeRecorder = GetTakeRecorder();
  386. if (takeRecorder != null)
  387. {
  388. var slate = takeRecorder.GetActiveSlate();
  389. TakeManager.Default.SetIterationBase(slate, guid);
  390. SendSlateDescriptor(slate);
  391. Refresh();
  392. }
  393. }
  394. void ClientClearIterationBase()
  395. {
  396. var takeRecorder = GetTakeRecorder();
  397. if (takeRecorder != null)
  398. {
  399. var slate = takeRecorder.GetActiveSlate();
  400. TakeManager.Default.ClearIterationBase(slate);
  401. SendSlateDescriptor(slate);
  402. Refresh();
  403. }
  404. }
  405. void OnTexturePreviewRequested(Guid guid)
  406. {
  407. var texture = TakeManager.Default.GetAssetPreview<Texture2D>(guid);
  408. if (texture != null && TryGetInternalClient(out var client))
  409. {
  410. client.SendTexturePreview(guid, texture);
  411. }
  412. }
  413. bool OnClientConnected(ICompanionAppClient client)
  414. {
  415. if (m_Client == null && client is TClient c && (!ClientMappingDatabase.TryGetClientAssignment(this, out var clientName, out _) || c.Name == clientName))
  416. {
  417. SetClient(c, false);
  418. return true;
  419. }
  420. return false;
  421. }
  422. void OnClientDisconnected(ICompanionAppClient client)
  423. {
  424. if (m_Client == client)
  425. {
  426. SetClient(null, false);
  427. }
  428. }
  429. }
  430. }

本人打包后成功运行,测试脚本:


  
  1. using UnityEngine;
  2. using Unity.LiveCapture.CompanionApp;
  3. public class LiveCaptureExample : MonoBehaviour
  4. {
  5. LiveCaptureServer server;
  6. private void Awake()
  7. {
  8. server = Resources.Load<LiveCaptureServer>("Live Capture Server");
  9. server.Init();
  10. server.StartServer();
  11. }
  12. private void Update()
  13. {
  14. server.OnUpdate();
  15. }
  16. private void OnDestroy()
  17. {
  18. server.StopServer();
  19. }
  20. }

文章来源: coderz.blog.csdn.net,作者:CoderZ1010,版权归原作者所有,如需转载,请联系作者。

原文链接:coderz.blog.csdn.net/article/details/123455444

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。